深入了解Java中的锁机制

目录

1. synchronized关键字

1.1 基本概念

1.2 内置锁

1.3 限制

2. ReentrantLock

2.1 概述

2.2 公平性与非公平性

2.3 条件变量

3. 读写锁(ReadWriteLock)

3.1 概念

3.2 适用场景

4. StampedLock

4.1 概述

4.2 乐观读与悲观读

4.3 适用场景

5. 性能比较与选择

6. 总结


        在Java编程中,多线程并发是一个常见的场景。为了保证线程安全性,Java提供了一系列的锁机制,用于控制对共享资源的访问。这些锁机制在并发编程中起着至关重要的作用,确保多个线程能够协同工作而不产生竞态条件或数据不一致的问题。本文将深入探讨Java中的锁机制,包括传统的synchronized关键字、ReentrantLock类以及更为高级的读写锁和StampedLock。

1. synchronized关键字

1.1 基本概念

        Java的synchronized关键字是最基本的锁机制之一。它可以用来修饰方法或代码块,确保同一时刻只有一个线程能够访问被锁定的代码。

public synchronized void synchronizedMethod() {
    // 线程安全的操作
}

1.2 内置锁

   synchronized使用的是内置锁,也称为监视器锁。每个Java对象都有一个与之关联的内置锁,通过synchronized关键字可以对这个锁进行操作。当一个线程试图访问一个被synchronized修饰的方法或代码块时,它会尝试获取对象的内置锁,如果锁已经被其他线程占用,那么线程将被阻塞,直到获取到锁为止。

1.3 限制

        虽然synchronized是简单易用的锁机制,但它也有一些限制。首先,它是非公平的,不能保证等待时间最长的线程会最先获得锁。其次,一旦线程进入synchronized代码块,其他线程必须等待,不能中途取消。

2. ReentrantLock

2.1 概述

   ReentrantLock是Java.util.concurrent包中提供的一种更灵活的锁机制。与synchronized不同,ReentrantLock允许线程在获得锁之后再次进入同步代码块,即支持重入。

import java.util.concurrent.locks.ReentrantLock;

public class ReentrantLockExample {
    private final ReentrantLock lock = new ReentrantLock();

    public void someMethod() {
        lock.lock();
        try {
            // 线程安全的操作
        } finally {
            lock.unlock();
        }
    }
}

2.2 公平性与非公平性

   ReentrantLock提供了公平性选择。在构造函数中,可以选择是否使用公平锁。公平锁按照线程请求锁的顺序进行获取,而非公平锁允许插队,可能会导致某些线程一直获取不到锁。

ReentrantLock fairLock = new ReentrantLock(true); // 公平锁
ReentrantLock unfairLock = new ReentrantLock();     // 非公平锁

2.3 条件变量

   ReentrantLock还支持条件变量,可以通过newCondition方法创建。条件变量允许线程在获取锁之后等待或者唤醒,提供了更为灵活的线程通信方式。

import java.util.concurrent.locks.Condition;

public class ReentrantLockWithCondition {
    private final ReentrantLock lock = new ReentrantLock();
    private final Condition condition = lock.newCondition();

    public void await() throws InterruptedException {
        lock.lock();
        try {
            condition.await();
        } finally {
            lock.unlock();
        }
    }

    public void signal() {
        lock.lock();
        try {
            condition.signal();
        } finally {
            lock.unlock();
        }
    }
}

3. 读写锁(ReadWriteLock)

3.1 概念

   ReadWriteLock接口提供了一种更为精细的锁分离机制,分为读锁和写锁。读锁可以被多个线程同时持有,但写锁是独占的,只能被一个线程持有。

import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;

public class ReadWriteLockExample {
    private final ReadWriteLock readWriteLock = new ReentrantReadWriteLock();

    public void readMethod() {
        readWriteLock.readLock().lock();
        try {
            // 执行读操作
        } finally {
            readWriteLock.readLock().unlock();
        }
    }

    public void writeMethod() {
        readWriteLock.writeLock().lock();
        try {
            // 执行写操作
        } finally {
            readWriteLock.writeLock().unlock();
        }
    }
}

3.2 适用场景

   ReadWriteLock适用于读多写少的场景,可以提高系统的并发性能。读锁的共享特性使得多个线程可以同时读取共享资源,而写锁的独占特性保证了写操作的原子性。

4. StampedLock

4.1 概述

   StampedLock是Java 8引入的新锁机制,结合了读写锁和乐观锁的特点。它引入了"stamp"的概念,用来标记锁的状态。

import java.util.concurrent.locks.StampedLock;

public class StampedLockExample {
    private final StampedLock stampedLock = new StampedLock();

    public void readMethod() {
        long stamp = stampedLock.tryOptimisticRead();
        try {
            // 乐观读操作
            if (!stampedLock.validate(stamp)) {
                // 转为悲观读
                stamp = stampedLock.readLock();
                try {
                    // 执行悲观读操作
                } finally {
                    stampedLock.unlockRead(stamp);
                }
            }
        } finally {
            stampedLock.unlock(stamp);
        }
    }

    public void writeMethod() {
        long stamp = stampedLock.writeLock();
        try {
            // 执行写操作
        } finally {
            stampedLock.unlockWrite(stamp);
        }
    }
}

4.2 乐观读与悲观读

   StampedLock引入了乐观读和悲观读的概念。在乐观读模式下,线程尝试获取一个标记(stamp),然后进行读操作,最后通过validate方法验证标记是否仍然有效。如果标记无效,表示在读操作期间有写操作发生,需要切换为悲观读模式。悲观读模式下,线程直接获取读锁,执行读操作,然后释放读锁。

4.3 适用场景

   StampedLock适用于读操作远远多于写操作的情况,并且乐观读是常态的场景。相较于ReadWriteLockStampedLock提供了更高的并发性能。

5. 性能比较与选择

        在选择锁的时候,需要根据具体的业务场景和性能需求来进行权衡。以下是一些选择锁的一些建议:

  • 如果并发要求不高,可以使用synchronized关键字,它简单易用,不需要手动释放锁,适用于简单的线程同步场景。

  • 如果需要更灵活的控制和可重入特性,可以选择ReentrantLock,并且可以根据实际情况选择公平锁或非公平锁。

  • 如果读操作远远多于写操作,可以选择ReadWriteLock,提高系统的并发性能。

  • 如果乐观读是常态,并且读操作频繁,可以考虑使用StampedLock,它提供了更高的并发性能。

6. 总结

        Java中的锁机制为多线程编程提供了强大的支持,开发人员可以根据实际需求选择合适的锁来保证线程安全性。从简单的synchronized关键字到更为灵活的ReentrantLock,再到适用于读多写少场景的ReadWriteLock,以及引入了乐观读的StampedLock,Java提供了丰富的锁机制,帮助开发人员更好地处理并发编程中的各种情况。在实际应用中,合理选择锁机制是提高系统性能和稳定性的关键一步。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/260411.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

仓储1代电子标签接口文档

标签注册 仓储1代注册 侧面按钮连按三次, 注册成功:红灯变绿灯 查询电子标签信息接口 接口地址:192.168.1.200/wms/associate/getTagsMsg 请求类型:multipart/form-data 请求方式:get 接口备注:暂无描…

如何设计用户评论表

作者简介:大家好,我是smart哥,前中兴通讯、美团架构师,现某互联网公司CTO 联系qq:184480602,加我进群,大家一起学习,一起进步,一起对抗互联网寒冬 上一篇提到树形结构是非…

C++_动态二维数组的两种方法

介绍 本文主要介绍使用 动态二维数组的两种方法 (PS:仅作创建 动态二维数组参考,详细使用方法根据需求自行改变) 第一种:连续存储结构的 二维动态数组(需固定 列 大小,可通过下标访问) 缺点: 1.需要在设计二维数组前写死 列 的大小 2.空间利用率不高 优点…

netty源码:(29)ChannelInboundHandlerAdapter

它实现的方法都有一个ChannelHandlerContext参数,它的方法都是直接调用ChannelHandlerContext参数对应的方法,该方法会调用下一个handler对应的方法。 可以继承这个类,重写感兴趣的方法,比如channelRead. 这个类有个子类:SimpleC…

调度算法(一)-第二十一天

目录 各种调度算法的学习思路 先来先服务(FCFS) 短作业优先(SJF) 各种调度算法的学习思路 1、算法思想 2、算法规则 3、算法用于作业调度还是进程调度 5、抢占式还是非抢占式 6、优点和缺点 7、是否会导致饥饿(进程…

【音视频 | AAC】AAC格式音频文件解析

😁博客主页😁:🚀https://blog.csdn.net/wkd_007🚀 🤑博客内容🤑:🍭嵌入式开发、Linux、C语言、C、数据结构、音视频🍭 🤣本文内容🤣&a…

C++刷题 -- KMP算法

C刷题 – KMP算法 文章目录 C刷题 -- KMP算法1.算法讲解2.算法实现 https://leetcode.cn/problems/find-the-index-of-the-first-occurrence-in-a-string/description/ 1.算法讲解 KMP算法是一种字符串匹配算法,当出现字符串不匹配时,可以记录一部分之…

PSP - 结构生物学中的机器学习 (NIPS MLSB Workshop 2023.12)

欢迎关注我的CSDN:https://spike.blog.csdn.net/ 本文地址:https://spike.blog.csdn.net/article/details/135120094 Machine Learning in Structural Biology (机器学习在结构生物学中) 网址:https://www.mlsb.io/ Workshop at the 37th Co…

计算机网络-进阶

目录 易混淆物理层数据链路层网络层nat如何实现私有ip通信IP数据报 格式解析tcp 连接tcp流量控制滑动窗口拥塞控制 报文捕获 wireshark路由模拟器 enspcdn代理服务器 VS cdn VS web cache 计算机有了物理地址,为什么还要有ip地址?单播 多播 广播 传输层会…

基于Java+SpringBoot+Mybaties-plus+Vue+ElementUI+Vant 电影院订票管理系统 的设计与实现

一.项目介绍 基于SpringBootVue 电影院订票管理系统 分为前端和后端。 前端(用户): 登录后支持查看首页、电影、影院和我的信息 支持查看正在热映和即将上映的电影信息 支持购票(需选择影院座位)、看过(评论…

力扣:203. 移除链表元素(Python3)

题目: 给你一个链表的头节点 head 和一个整数 val ,请你删除链表中所有满足 Node.val val 的节点,并返回 新的头节点 。 来源:力扣(LeetCode) 链接:力扣(LeetCode)官网 …

基于springboot+mybatis+mysql+jsp房屋租赁管理系统

基于springbootmybatismysqljsp房屋租赁管理系统 一、系统介绍二、功能展示1.项目内容2.项目骨架3.数据库3.登录4.首页5.房源管理6.个人中心7.房屋详情 四、其它1.其他系统实现五.获取源码 一、系统介绍 项目名称:基于Spring boot的房屋租赁管理系统 项目架构&…

对77,539个基因组进行的遗传关联分析揭示了罕见疾病的病因

今天给同学们分享一篇实验文章“Genetic association analysis of 77,539 genomes reveals rare disease etiologies”,这篇文章发表在Nat Med期刊上,影响因子为82.9。 结果解读: 稀有水库 关系型数据库(RDB)提供了一…

vue关闭当前路由页面并跳转到其父页面

1.dom中添加关闭或取消按钮 <el-button type"primary" class"blueLinearbg cancelBtn" click"cancel" >取 消</el-button>2.cancel方法中 /*取消或关闭*/cancel(){this.$store.dispatch("tagsView/delView", this.$route)…

echarts 实现x轴文字倾斜显示

显示效果 关键代码 xAxis: {axisLabel: {show: true,rotate: 35,//35度角倾斜显示},}, 完全代码 var optSaleType {title: {text: ,textStyle: {color: #000,fontSize: 14}},tooltip: {},grid: {left: 0,right: 0,bottom: 0,containLabel: true,},xAxis: {axisLabel: {show:…

苹果cms论坛多播放源自动采集 /采集在线影视网站/苹果CMS影视站采集器

源码介绍&#xff1a; 苹果cms论坛多播放源自动采集、采集在线影视网站&#xff0c;作为苹果CMS影视站采集器&#xff0c;它能轻松获取在线影视网站资源。 苹果 cms 论坛这是一个基于Vue和Gin实现的在线观影网站。项目采用 vite vue 作为前端技术栈, 使用 ElementPlus 作为 …

基于mpvue的小程序项目搭建的步骤(附精选源码32套,涵盖商城团购等)

mpvue 是美团开源的一套语法与vue.js一致的、快速开发小程序的前端框架&#xff0c;按官网说可以达到小程序与H5界面使用一套代码。使用此框架&#xff0c;开发者将得到完整的 Vue.js 开发体验&#xff0c;同时为 H5 和小程序提供了代码复用的能力。如果想将 H5 项目改造为小程…

ES排错命令

GET _cat/indices?v&healthred GET _cat/indices?v&healthyellow GET _cat/indices?v&healthgreen确定哪些索引有问题&#xff0c;多少索引有问题。_cat API 可以通过返回结果告诉我们这一点 查看有问题的分片以及原因。 这与索引列表有关&#xff0c;但是索引…

海康rtsp拉流,rtmp推流,nginx部署转flv集成

海康rtsp拉流&#xff0c;rtmp推流&#xff0c;nginx部署转flv集成 项目实际使用并测试经正式使用无问题&#xff0c;有问题欢迎评论留言 核心后台java代码&#xff1a; try {// FFmpeg命令String command "ffmpeg -re -i my_video.mp4 -c copy -f flv rtmp://localho…

opencv入门到精通——鼠标事件和Trackbar控件的使用

目标 了解如何在OpenCV中处理鼠标事件 您将学习以下功能&#xff1a;cv.setMouseCallback() 了解将轨迹栏固定到OpenCV窗口 您将学习以下功能&#xff1a;cv.getTrackbarPos&#xff0c;cv.createTrackbar等。 简单演示 在这里&#xff0c;我们创建一个简单的应用程序&am…