Java中的synchronized关键字

目录

1、synchronized是什么

2、synchronized的用法

synchronized可以用在方法或者代码块上,分别称为同步方法和同步代码块。

用法理解

3、synchronized的实现原理

⭐synchronized锁的对比

4、synchronized的优缺点

⭐扩展:synchronized 和 volatile 的区别?

⭐扩展:synchronized与Lock的区别?

小结


1、synchronized是什么

`synchronized` 是Java中的一个关键字,用于实现线程间同步。它可以被用来修饰方法或代码块,使得同一时刻只有一个线程可以执行被 `synchronized` 修饰的代码。

在Java中,当多个线程并发执行时,可能会出现数据竞争和不一致的情况。为了避免这种情况的发生,我们需要对共享变量进行同步控制,以保证同一时刻只有一个线程能够访问共享变量。

具体地说,当一个线程进入一个被 `synchronized` 修饰的方法或代码块时,它会尝试获取这个方法或代码块所属对象的锁(也称为监视器锁),如果获取不到锁就会阻塞等待。当另一个线程执行完该方法或代码块并释放了锁之后,等待的线程才能获取到锁并继续执行。

`synchronized` 的使用可以有效地避免多个线程同时访问共享变量造成的问题,但也会引入一定的性能损耗。因此,在需要进行线程同步时,建议使用 `synchronized` 来实现。但在高并发场景下,也可以考虑使用更轻量级的锁,例如 `ReentrantLock` 或 `Atomic` 类。

图片来源:https://www.cnblogs.com/three-fighter/p/14396208.html

2、synchronized的用法

synchronized可以用在方法或者代码块上,分别称为同步方法和同步代码块。

如果修饰的是普通方法,则锁作用于当前对象实例。如果是修饰静态方法,锁作用于类的Class实例。如果修饰的是代码块,作用于当前对象实例,则需要指定加锁对象。

1. 同步代码块:使用 synchronized 关键字修饰的代码块,可以指定一个对象作为锁,只有获得该锁的线程才能执行代码块。

synchronized后面括号里是一对象,此时,线程获得的是对象锁.例如:

public classMyThread implements Runnable{public static void main(Stringargs[]){MyThread mt=new MyThread();Thread t1=newThread(mt,"t1");Thread t2=newThread(mt,"t2");Thread t3=newThread(mt,"t3");Thread t4=newThread(mt,"t4");Thread t5=newThread(mt,"t5");Thread t6=newThread(mt,"t6");t1.start();t2.start();t3.start();t4.start();t5.start();t6.start();
}   
public void run(){synchronized(this){System.out.println(Thread.currentThread().getName());}
}
public void method() {  synchronized (lockObject) { //一次只能有一个线程进入 // 执行的代码  }  
}

在上述代码中,lockObject 是一个任意的对象,它将作为锁来确保同步。只有获取到 lockObject 对象的锁的线程才能执行同步代码块。

2. 同步方法:使用 synchronized 关键字修饰的方法,整个方法都会被视为同步代码块,同一时间只允许一个线程执行该方法。

public synchronized void synchronizedMethod() {// 需要同步的方法体
}

其中,锁对象可以是任意对象,只要在多个线程间能保持唯一性即可。通常,我们使用被访问对象的引用作为锁对象,以保证同一时刻只有一个线程可以访问该对象的相关操作。

需要注意的是,同步方法的锁是当前对象实例(即 this),而同步代码块可以指定任意的对象作为锁。在使用 synchronized 关键字时,需要选择合适的锁对象来确保线程安全,并避免死锁和性能问题。

用法理解

图片来源:synchronized(Java语言的关键字)_百度百科

3、synchronized的实现原理

它的实现原理主要基于Java 对象头、Monitor(监视器)以及对象的状态机等概念。当一个线程想要执行同步方法或同步块时,它必须先获取该方法或块的锁。如果其他线程已经持有该锁,那么当前线程就会进入阻塞状态,直到其他线程释放了它所持有的锁。这种机制可以避免多个线程同时访问共享资源,从而保证数据的一致性和安全性。

Java 对象头:在 Java 对象的内存布局中,每个对象都有一个头部信息。对象头是对象实例的一部分。它包含了对象的元数据信息,如对象的哈希码、锁状态标志等。在synchronized实现中,对象头被用来作为锁的标识。当一个线程执行synchronized方法时,它需要获取该方法所在对象的对象头锁。如果其他线程已经持有该锁,那么当前线程就会进入阻塞状态,直到其他线程释放了它所持有的锁。

图片来源:https://www.cnblogs.com/three-fighter/p/14396208.html

在Java中,synchronized的实现是通过对象头中的Mark Word来实现的。Mark Word是Java对象头中的一个重要组成部分,它包含了对象的哈希码、类型信息、锁状态等信息。当一个线程执行synchronized方法时,JVM会通过CAS(Compare and Swap)操作来尝试获取该对象的锁。如果获取成功,那么该线程就可以执行synchronized方法;如果获取失败,那么该线程就会进入阻塞状态。

有关CAS的内容可以查看博客:http://t.csdnimg.cn/8QSh6

64 位虚拟机 Mark Word 是 64bit,在运行期间,Mark Word里存储的数据会随着锁标志位的变化而变化。

图片来源:https://www.cnblogs.com/three-fighter/p/14396208.html

32位?

图片来源:synchronized 看这一篇就够了 - 知乎

Monitor(监视器):每个 Java 对象都与一个 Monitor 相关联,Monitor 是用来实现对象的锁机制的一种数据结构。它包含了锁的拥有者线程、等待队列、计数器等信息。

对象的状态机:Java 对象在并发环境下可以处于不同的状态,如无锁状态、偏向锁状态、轻量级锁状态和重量级锁状态等。

当执行 synchronized 修饰的方法或代码块时,根据对象的状态,JVM 会进行如下处理:

1. 无锁状态:当对象没有被任何线程锁定时,进入 synchronized 代码块的线程将会尝试获取对象的锁。
2. 偏向锁状态:如果对象的锁处于无锁状态且没有竞争,那么进入 synchronized 代码块的线程可以直接获取锁,并将对象头中的线程ID更新为自己的ID,此时对象处于偏向锁状态。
3. 轻量级锁状态:如果对象处于偏向锁状态但出现了竞争,JVM 会尝试使用轻量级锁来实现同步。它通过CAS(比较并交换)操作来尝试获取锁,如果获取成功,则执行 synchronized 代码块;否则进入重量级锁状态。
4. 重量级锁状态:当多个线程争用同一个对象的锁时,JVM 会将对象的状态升级为重量级锁状态,此时线程会被阻塞,并加入到对象的等待队列中。只有拥有锁的线程释放锁后,等待队列中的线程才能被唤醒。

synchronized 看这一篇就够了 - 知乎

无论是偏向锁、轻量级锁还是重量级锁,它们都是通过在对象头中设置标记位和指针来实现的。JVM 会根据对象的竞争情况自动选择适合的锁状态,并进行状态的转换。

需要注意的是,synchronized 关键字的实现细节可能因不同的 JVM 实现而有所差异,上述描述是基于经典的 HotSpot JVM。

⭐synchronized锁的对比

优点缺点使用场景
偏向锁加锁和解锁不需要CAS操作,没有额外的性能消耗,和执行非同步方法相比仅存在纳秒级的差距如果线程间存在锁竞争,会带来额外的由于锁撤销的消耗适用于只有一个线程访问同步块的场景
轻量级锁竞争的线程不会阻塞,提高了响应速度如果线程一直得不到锁竞争的线程,使用自旋会消耗CPU性能追求响应时间,同步块执行速度非常快
重量级锁线程竞争不适用自旋,不会消耗CPU线程阻塞,响应时间缓慢,在多线程下,频繁的获取释放锁,带来的性能消耗很大追求吞吐量,同步块执行速度较长

4、synchronized的优缺点

优点
1. 简单易用:`synchronized` 关键字的语法简单,易于理解和使用,可以方便地确保多个线程对共享资源的安全访问。
2. 内置支持:作为Java语言的一部分,`synchronized` 关键字得到了JVM层面的支持,避免了用户自行实现线程同步机制的复杂性。
3. 可重入性:synchronized锁是可重入的,一个线程可以多次获得同一个锁,而不会造成死锁。

缺点
1. 粒度粗:使用 synchronized 关键字进行同步时,通常是对整个方法或代码块进行同步,这可能会导致一些不必要的等待,降低并发性能。
2. 无法中断:一旦进入 synchronized 代码块,除非获取到锁否则无法被中断,这可能会导致线程挂起的时间过长。
3. 性能开销:在某些情况下,使用 synchronized 可能会引入一定的性能开销,特别在高并发的场景下,这种开销可能会更加显著。
4. 局限性:synchronized 的锁是基于对象的,因此如果需要对不同的资源进行管理,就需要创建不同的对象锁,这可能会增加复杂性。

总的来说,`synchronized` 是一种简单且有效的线程同步机制,但在一些特定的情况下可能存在一些性能和灵活性上的局限性。在实际开发中,可以根据具体情况选择合适的同步机制,例如 `ReentrantLock`、`ReadWriteLock` 等来弥补 `synchronized` 的不足。

⭐扩展:synchronized 和 volatile 的区别?

synchronized 和 volatile 都是 Java 中用于保证多线程程序正确性的关键字,虽然它们的作用有所不同,但可以作为互补。

synchronized 关键字用于实现原子性操作和互斥访问。使用 synchronized 修饰的代码块或方法,在同一时间只允许一个线程进入临界区,其他线程需要等待当前线程执行完毕后才能进入。因此,synchronized 能够保证多个线程对共享资源的安全访问,并防止数据竞争和不一致性。

volatile 关键字用于保证可见性和禁止指令重排序。使用 volatile 修饰的变量,在多个线程之间保持可见性,当一个线程修改了变量的值,其他线程能够立即看到最新的值。此外,volatile 还能够禁止编译器和处理器对代码的优化,确保指令按照程序的顺序执行,避免出现意外的结果。

需要注意的是,volatile 不能保证原子性,如果需要进行复合操作,例如自增、自减、比较并交换等,仍然需要使用 synchronized 或 Lock 等机制来确保原子性。
 

⭐扩展:synchronized与Lock的区别?

图片来源:详解synchronized与Lock的区别与使用_synchronized 与lock 和redissionclient分布式锁区别-CSDN博客

小结

注意:在 JDK1.5之前synchronized是一个重量级锁,相对于juc包中的Lock,synchronized显得比较“重量级”。但是在 Java 6 之后 Java 官⽅对从 JVM 层⾯对synchronized进行了优化,例如“偏向锁”、“轻量锁”等等,并作为Java并发场景下实现多线程安全的一种比较直接的操作。

参考:

synchronized(Java语言的关键字)_百度百科

https://www.cnblogs.com/three-fighter/p/14396208.html

你真的了解 Synchronized 吗? - 知乎

synchronized 看这一篇就够了 - 知乎

详解synchronized与Lock的区别与使用_synchronized 与lock 和redissionclient分布式锁区别-CSDN博客
https://www.cnblogs.com/aspirant/p/11470858.html
 

万字干货|Synchronized关键字详解 - 知乎

深入Synchronized各种使用方法 - 一无是处的研究僧 - 博客园


感谢阅读,码字不易,多谢点赞!如有不当之处,欢迎反馈指出,感谢!

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

nvm安装管理nodejs版本

1:如果之前先安装了nodejs先卸载nodejs 2:下载nvm,点击下载路径https://github.com/coreybutler/nvm-windows/releases,选择相应环境下载,如下window环境下载 下载成功后,选择NVM安装在哪个文件目录下&…

IDEA之Eclipse Code Formatter插件的安装与使用

概述 Eclipse、Intellij idea格式化结果不同,之前由于没有Eclipse Code Formatter插件,所以公司统一用eclipse做开发,但是我们都知道IDEA是非常强大、也非常的方便,很多功能是eclipse不具备的,只是我们公司统一用&…

【隐私计算】VOLE (Vector Oblivious Linear Evaluation)学习笔记

近年来,VOLE(向量不经意线性评估)被用于构造各种高效安全多方计算协议,具有较低的通信复杂度。最近的CipherGPT则是基于VOLE对线性层进行计算。 1 VOLE总体设计 VOLE的功能如下,VOLE发送 Δ \Delta Δ和 b b b给send…

【论文阅读笔记】InstructDiffusion: A Generalist Modeling Interface for Vision Tasks

【论文阅读笔记】StyleAvatar3D: Leveraging Image-Text Diffusion Models for High-Fidelity 3D Avatar Generation 论文阅读笔记论文信息引言动机挑战 方法结果 关键发现相关工作1. 视觉语言基础模型2. 视觉通用模型 方法/模型视觉任务的统一说明训练数据构建网络结构 实验设…

全网最新最全面的Jmeter接口测试:jmeter模拟http请求实战

1、get请求 http://www.hnxmxit.com/ 2、带参数的get请求 微信公众号获取token请求 3、自定义头部信息的请求 百度搜索请求 https://www.baidu.com/s?wd猫 4、post请求 微信公众号添加用户标签请求 注:post请求中如果body中的数据为json,一定要在信息头管理器中…

编译原理头歌实验:实验1《词法分析程序设计与实现》(C语言版)

任务描述 本关任务:加深对词法分析器的工作过程的理解;加强对词法分析方法的掌握;能够采用一种编程语言实现简单的词法分析程序;能够使用自己编写的分析程序对简单的程序段进行词法分析。 相关知识 为了完成本关任务&#xff0…

Qt应用开发--国产工业开发板全志T113-i的部署教程

Qt在工业上的使用场景包括工业自动化、嵌入式系统、汽车行业、航空航天、医疗设备、制造业和物联网应用。Qt被用来开发工业设备的用户界面、控制系统、嵌入式应用和其他工业应用,因其跨平台性和丰富的功能而备受青睐。 Qt能够为工业领域带来什么好处: -…

袋鼠云产品功能更新报告08期|近百项全新功能和优化,你要的都在这里!

欢迎来到袋鼠云08期产品功能更新报告!在瞬息万变的市场环境中,我们深知客户的需求与期待,因此,我们及时推出袋鼠云最新产品更新及优化,包括数据治理中心、Hive SQL 性能优化、新插件等,助力企业在数字世界中…

Golang分布式事务

引言 在分布式系统中,事务管理是一项非常重要的任务。分布式事务涉及到多个事务参与者之间的协调和一致性保证,同时还要解决网络延迟、故障恢复等问题。Golang作为一门强大的编程语言,提供了一些工具和框架来帮助开发人员实现分布式事务。本…

awk从放弃到入门(11):拾遗之”三元运算”与”打印奇偶行”

awk从放弃到入门(11):拾遗之”三元运算”与”打印奇偶行” 三元运算打印奇偶行 本博文转载自 这篇文章中的知识点是建立在前文的基础上的,如果你还没有掌握前文中的知识,请先参考之前的文章。 这篇文章其实是对之前知…

英语翻译小软件 ← Python实现

【程序描述】 利用Python实现一个英语翻译小软件。 ★ 当输入一个英文单词后,输出对应的中文意思。 ★ 当输入 q 时,退出程序。 ★ 当输入一个不存在的词条时,捕获异常,提示“No finding!”。【程序代码】 dict{&quo…

IOS/安卓+charles实现抓包(主要解决证书网站无法打开问题)

安装 官网下载 https://www.charlesproxy.com/latest-release/download.do 安装charles文档 流程 上述链接解决下图问题 使用介绍 Charles介绍 上述链接看一至三即可,了解首页各个按钮的作用 charles全面使用教程及常见功能详解(较详细&#xff09…

Vim编辑器使用

替换全部 :%s/search_string/replace_string/g 撤销 u 撤销 ctrl r 取消撤销 换行 下移动一行 -上移动一行 删除 插入模式删除字符 Ctrl U

netty源码:(1)NioEventLoopGroup

EventLoopGroup bossGroup new NioEventLoopGroup(); 不加参数创建NioEventLoopGroup的话,会使用cpu核数*2作为bossGroup的线程数。

二 使用GPIO的复用功能 利用USART 实现printf()

参考这篇: STM32串口通信详解 1. 关于USART USART ( universal synchronous / asynchronous receiver /transmitter) 是一种串行通讯协议 , 允许设备通过串行端口进行数据传输, USART 能够以同步或者异步的方式进行工作,在实际的运用中&…

容器技术发展史,编排与容器的技术演进之路——2

目录: 容器技术发展史 Jail时代 1979 年 贝尔实验室发明 chroot2000 年 FreeBSD 4.0 发行 FreeBSD Jail2001 年 Linux VServer 发行2004 年 Solaris Containers 发行云时代 2006 年 google 推出 Process Containers2008 年 LXC 推出2011 年 CloudFoundry 推出 Ward…

【Linux】I/O多路转接技术

I/O多路转接技术 一、I/O多路转接之select1、select函数2、fd_set的相关内容3、如何在代码中高效的使用select函数4、select服务器5、select的优缺点6、select的适用场景 二、I/O多路转接之poll1、poll函数2、struct pollfd结构2、poll服务器3、poll的优缺点 三、I/O多路转接之…

自身免疫疾病抗原应用方向——博迈伦生物

自身免疫疾病是指机体免疫系统对自身组织产生异常免疫反应的疾病,包括类风湿性关节炎、系统性红斑狼疮、多发性硬化症等。目前,针对自身免疫疾病的治疗方案主要包括药物治疗和免疫调节治疗。而自身免疫疾病抗原应用则是一种新的治疗方向,其通…

Mac IDEA解决Maven项目命令行报错:command not found: mvn

1. 使用idea自带的maven命令 open -e ~/.zshrc 2. 在其最下面增加 # maven export MAVEN_HOME"/Applications/IntelliJ IDEA.app/Contents/plugins/maven/lib/maven3" export PATH$MAVEN_HOME/bin:$PATH # maven end 3. 连接使之生效 source ~/.zshrc4. 修改mvn…

Promise介绍

前言 Promise是一种用于处理异步操作的JavaScript对象,它可以将生产代码和消费代码连接起来,让异步方法像同步方法一样返回值。 异步操作是指在执行过程中可能会遇到延迟或等待的操作,例如网络请求、定时器、文件读取等。传统的JavaScript代码…