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.hqwc.cn/news/234969.html

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

相关文章

NSSCTF第14页(2)

[UUCTF 2022 新生赛]ezpop 提示说看看反序列化字符串逃逸 PHP反序列化字符串逃逸_php反序列化逃逸-CSDN博客 php反序列化字符逃逸_php反序列化逃逸_Leekos的博客-CSDN博客 buuctf刷题9 (反序列化逃逸&shtml-SSI远程命令执行&idna与utf-8编码漏洞)_extract($_post);…

JeecgBoot低代码开发—Vue3版前端入门教程

JeecgBoot低代码开发—Vue3版前端入门教程 后端接口配置VUE3 必备知识1.vue3新特性a. https://v3.cn.vuejs.org/b.setup的用法c.ref 和 reactive 的用法d.新版 v-model 的用法e.script setup的用法 2.TypeScript基础 后端接口配置 如何修改后台项目路径 http://127.168.3.52:8…

MySQL处理并发访问和高负载的关键技术和策略

我深知在数据库管理中处理并发访问和高负载的重要性。在这篇文章中,我将探讨MySQL处理并发访问和高负载的关键技术和策略,以帮助读者更好地优化数据库性能。 图片来源:MySQL处理并发访问和高负载的关键技术和策略 MySQL数据库在处理并发访问…

【合集】MQ消息队列——Message Queue消息队列的合集文章 RabbitMQ入门到使用

前言 RabbitMQ作为一款常用的消息中间件,在微服务项目中得到大量应用,其本身是微服务中的重点和难点。本篇博客是Message Queue相关的学习博客文章的合集篇,目前主要是RabbitMQ入门到使用文章,后续会扩展其他MQ。 目录 前言一、R…

ssm+vue的公司安全生产考试系统(有报告)。Javaee项目,ssm vue前后端分离项目。

演示视频: ssmvue的公司安全生产考试系统(有报告)。Javaee项目,ssm vue前后端分离项目。 项目介绍: 采用M(model)V(view)C(controller)三层体系结…

【OJ比赛日历】快周末了,不来一场比赛吗? #12.02-12.08 #15场

CompHub[1] 实时聚合多平台的数据类(Kaggle、天池…)和OJ类(Leetcode、牛客…)比赛。本账号会推送最新的比赛消息,欢迎关注! 以下信息仅供参考,以比赛官网为准 目录 2023-12-02(周六) #4场比赛2023-12-03…

.NET开源的处理分布式事务的解决方案

前言 在分布式系统中,由于各个系统服务之间的独立性和网络通信的不确定性,要确保跨系统的事务操作的最终一致性是一项重大的挑战。今天给大家推荐一个.NET开源的处理分布式事务的解决方案基于 .NET Standard 的 C# 库:CAP。 CAP项目介绍 CA…

11.30 C++类特殊成员函数

#include <iostream>using namespace std; class Per { private:string name;int age;double *high;double weight; public://构造函数Per(string name,int age,double high,double weight):name(name),age(age),high(new double(high)),weight(weight){cout << &q…

信贷专员简历模板

这份简历内容&#xff0c;以信贷专员招聘需求为背景&#xff0c;我们制作了1份全面、专业且具有参考价值的简历案例&#xff0c;大家可以灵活借鉴。 信贷专员简历在线编辑下载&#xff1a;百度幻主简历 求职意向 求职类型&#xff1a;全职 意向岗位&#xff1a;信贷专员 …

python pytorch实现RNN,LSTM,GRU,文本情感分类

python pytorch实现RNN,LSTM&#xff0c;GRU&#xff0c;文本情感分类 数据集格式&#xff1a; 有需要的可以联系我 实现步骤就是&#xff1a; 1.先对句子进行分词并构建词表 2.生成word2id 3.构建模型 4.训练模型 5.测试模型 代码如下&#xff1a; import pandas as pd im…

2021年9月15日 Go生态洞察:TLS加密套件的自动排序机制

&#x1f337;&#x1f341; 博主猫头虎&#xff08;&#x1f405;&#x1f43e;&#xff09;带您 Go to New World✨&#x1f341; &#x1f984; 博客首页——&#x1f405;&#x1f43e;猫头虎的博客&#x1f390; &#x1f433; 《面试题大全专栏》 &#x1f995; 文章图文…

指数退避和抖动

目录 引入 OCC 添加退避机制 添加抖动机制 小结 引入 OCC 乐观并发控制&#xff08;Optimistic Concurrency Control&#xff0c;OCC&#xff09;是一种既能保证多个写入者安全地修改单个对象又能避免丢失写入的古老方法OCC具有三个优点&#xff1a;只要底层存储可用&#…