怎么使用synchronized
这篇文章主要讲解了"怎么使用synchronized ",文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习"怎么使用synchronized "吧!
话说 synchronized
一、前言
说起java的锁呀,我们先想到的肯定是synchronized[ˈsɪŋ krə naɪ zd]了 ,这个单词很拗口,会读这个单词在以后的面试中很加分(我面试过一些人 不会读 ,他们说的是syn开头那个单词),不会读略显不专业,不过问题不大,会用,懂原理才是最重要的。
内容会由简入难,有时候可以放弃一部分难的东西。 标记一下 回头再看 可能更加明朗
二、DEMO
废话不多说,先写hello world !!
例子: 小强 和 小明 同居了,但是只有一个厕所,他们每天必做的事情就是抢坑位,那么用代码实现要怎么写呢 ?
/** * @author 木子的昼夜 ** 人 实体类 */public class Person { // 名字 private String name; // 上厕所 public void gotoWc() { Wc.useWc(this); } public String getName() { return name; } public void setName(String name) { this.name = name; }}/** * @author 木子的昼夜 * * 厕所 实体类 */public class Wc { /** * 使用厕所方法 * @param p 使用厕所的人 */ public static void useWc(Person p){ try{ System.out.println(p.getName()+" 正在使用厕所!!"); TimeUnit.SECONDS.sleep(10); System.out.println(p.getName()+" 用完了!!"); } catch (Exception e) { // 厕所万一坏了 也得结束使用 System.out.println(p.getName()+" 用完了!!"); } }}/** * @author 木子的昼夜 * 这个测试 是小强与小明商量好了 说小强你先来 小强完事儿了 小明再来 * 这个不会发生冲突 因为是商量好的 顺序执行 * 大家都知道 顺序是不会出什么问题的 */public class SyncTest { public static void main(String[] args) { // 小强对象 Person xiaoqiang = new Person(); xiaoqiang.setName("小强"); // 小明对象 Person xiaoming = new Person(); xiaoming.setName("小明"); // 上厕所 xiaoqiang.gotoWc(); xiaoming.gotoWc(); }}
上厕所过程:
如果俩人没商量,自己去自己的呢 ?
/** * @author 木子的昼夜 */public class SyncTest02 { public static void main(String[] args) { // 小强对象 Person xiaoqiang = new Person(); xiaoqiang.setName("小强"); // 小明对象 Person xiaoming = new Person(); xiaoming.setName("小明"); // 开启两个线程 谁也不理谁 自己干自己的 new Thread(()->xiaoqiang.gotoWc()).start(); new Thread(()->xiaoming.gotoWc()).start(); }}
上厕所过程:
上图很明显可以看出来,小强没上完呢,小明就去上了,要是小的还凑活,大的怎么办? 画面自己想~~
这个时候大家可能想到了,厕所门上没锁吗? 谁先进去锁住不就行了吗?
答对了!
/** * @author 木子的昼夜 * * 改造后 厕所 实体类 */public class Wc { /** * 使用厕所方法 * synchronized: 谁先进厕所 马上上锁 !! * @param p 使用厕所的人 */ public static synchronized void useWc(Person p){ try{ System.out.println(p.getName()+" 正在使用厕所!!"); TimeUnit.SECONDS.sleep(10); System.out.println(p.getName()+" 用完了!!"); } catch (Exception e) { // 厕所万一坏了 也得结束使用 System.out.println(p.getName()+" 用完了!!"); } }}/** * @author 木子的昼夜 */public class SyncTest02 { public static void main(String[] args) { // 小强对象 Person xiaoqiang = new Person(); xiaoqiang.setName("小强"); // 小明对象 Person xiaoming = new Person(); xiaoming.setName("小明"); // 开启两个线程 谁也不理谁 自己干自己的 但是这次厕所有锁, // 谁先进去 就把锁锁住 new Thread(()->xiaoqiang.gotoWc()).start(); new Thread(()->xiaoming.gotoWc()).start(); }}
上厕所过程:
可以看到,小强先上完,小明再上的,这样就不会出什么问题了。
有人可能会问了,只见上锁,没有解锁,小明怎么进去的?
这就是synchronized的一个特性了,它会自动释放锁, synchronized包裹的代码执行完之后,锁就自动释放了。
所以避免了忘记释放锁,带来的尴尬~
三、 假装学术讨论
3.1 为什么要上锁?
以厕所为例,自己想去吧。 提出:"共享资源" 这个词就对了
3.2 对象锁 类锁
1) 对象锁 顾名思义 就是锁一个对象
/** * @author 木子的昼夜 * 对象锁 */public class SyncObject { /** * 累加值 (共享资源) */ int count = 0; /** * 锁对象 */ private Object lock = new Object(); public static void main(String[] args) { SyncObject so = new SyncObject(); // 线程1 new Thread(()->{ try { for (;;){ TimeUnit.SECONDS.sleep(2); so.increaseCount(); } } catch (Exception e) { System.err.println("错误"); } },"线程1").start(); // 线程2 new Thread(()->{ try { for (;;){ TimeUnit.SECONDS.sleep(2); so.increaseCount(); } } catch (Exception e) { System.err.println("错误"); } },"线程2").start(); } /** * count累加 */ public void increaseCount(){ // 加锁 synchronized (lock){ count = count+1; System.out.println(Thread.currentThread().getName()+" count="+count); } }}
例子中:两个线程增加count ,锁的是lock这个对象 ,这就叫对象锁 。
有时候会看见synchronized(this) 这是什么锁 ? this嘛 就是指当前对象,也是对象锁,
synchronized(this) 相当于 在方法上加synchronized,下边这两个方法都是锁的当前对象
/** * count累加 */ public void increaseCount(){ // 加锁 synchronized (this){ count = count+1; System.out.println(Thread.currentThread().getName()+" count="+count); } } /** * count累加 加锁 */ public synchronized void increaseCount02(){ count = count+1; System.out.println(Thread.currentThread().getName()+" count="+count); }
2) 类锁 顾名思义 就是给一个类加锁
每个类load到内存之后呢,会生成一个Class类型的对象 锁的就是他
其实也是锁一个对象 只是这个对象比较特殊,它代表类
/** * @author 木子的昼夜 * 对象锁 */public class SyncObject03 { /** * 累加值 (共享资源) */ static int count = 0; /** * 锁对象 */ private Object lock = new Object(); public static void main(String[] args) { SyncObject03 so = new SyncObject03(); // 线程1 new Thread(()->{ for (;;){ SyncObject03.increaseCount(); } },"线程1").start(); // 线程2 new Thread(()->{ for (;;){ SyncObject03.increaseCount(); } },"线程2").start(); } /** * count累加 */ public synchronized static void increaseCount(){ // 加锁 try { count = count+1; System.out.println(Thread.currentThread().getName()+" count="+count); TimeUnit.SECONDS.sleep(1); } catch (Exception e){ System.err.println("错误~"); } } /** * count累加 */ public void increaseCount02(){ synchronized(SyncObject03.class){ // 加锁 try { count = count+1; System.out.println(Thread.currentThread().getName()+" count="+count); TimeUnit.SECONDS.sleep(1); } catch (Exception e){ System.err.println("错误~"); } } }}
以上两种方式 都是类锁。
3.3 上锁方法执行的时候 可以执行当前对象未上锁方法吗?
这是一个用脚指头就能想到的答案,但是好多面试官问。 问了之后呢 你就蒙了~~ 难道不能?
答案是:能!
为什么能呢?因为爱所以爱~~ 错了,重来 .. 因为能所以能~~
小明在吃饭,给碗上个锁,别人不能用, 那小明能同时看他的偶像邓紫棋唱歌吗 ?
谁要是说不可以,以后吃饭不让他玩手机、pad、电脑 。 就让他吃吃吃 (那是猪)
/** * @author 木子的昼夜 * 可能出现的面试题 */public class SyncObject04 { private Object lock = new Object(); public static void main(String[] args) throws InterruptedException { SyncObject04 so = new SyncObject04(); // 线程1 new Thread(()->{ so.increaseCount(); },"线程1").start(); Thread.sleep(2000); // 线程2 new Thread(()->{ so.lookMv(); },"线程2").start(); } /** * 吃饭 */ public void increaseCount(){ // 加锁 synchronized (this){ try{ System.out.println("吃饭 "); // 也可以在吃饭这里 跟妹子聊天 chatWithGirl(); TimeUnit.SECONDS.sleep(10); } catch (Exception e) { System.err.println("饭掉地上了~"); } System.out.println("吃完饭 "); } } /** * 看演唱会视频 */ public void lookMv(){ System.out.println("看演唱会视频"); } /** * 看跟美女聊天 */ public void chatWithGirl(){ System.out.println("跟美女聊天"); }}
3.4 可重入
能一句话总结吗?
咳咳: 同一线程可以调用加了同一把锁的两个方法 不会阻塞。
例子:同一个人可以用同一双筷子(筷子加锁),吃不同的菜~~
吃鱼的时候获取了锁,在吃鱼方法里调用吃沙拉方法,是可以调用成功了 因为两个方法用的同一把锁
/** * @author 木子的昼夜 */public class TestC { /** * 一双筷子 只能一个人同一时间使用(一个线程) */ Object chopsticks = new Object(); public static void main(String[] args) { TestC c = new TestC(); new Thread(()->{ c.eatFish(); }).start(); } /** * 吃 */ public void eatFish( ) { synchronized (chopsticks) { try { System.out.println("吃 鱼"); Thread.sleep(2000); eatSalad(); } catch (Exception e){ } } } /** * 吃沙拉 */ public void eatSalad() { synchronized (chopsticks) { try { System.out.println("吃 沙拉"); Thread.sleep(2000); eatFish(); } catch (Exception e){ } } }}
3.5 底层实现
(1) 简单版
jdk1.6之前 synchronized 是 重量级锁 什么是重量级锁? 就是每次锁都会去找操作系统申请锁。
jdk1.6及以后改进为锁升级
简单思路是:
synchronized(object)
线程A 第一个访问
偏向锁 只在object的markword 中记录线程A的线程ID
如果线程A 又进来访问 一看markword的线程号是自己 那就直接用
这时候线程B 来了 ,线程B 一看我擦? 有人占用了锁!
线程B 会循环死等 ,类似在厕所门口,敲敲门问问线程A 你好了吗?敲敲门问问线程A 你好了吗?敲敲门问问线程A 你好了吗?敲敲门问问线程A 你好了吗?
线程B的这一操作,用术语将叫: 自旋锁
线程B 问10次之后,得不到锁,就会升级为重量级锁 (去操作系统申请资源)
无锁->偏向锁->自旋锁->重量级锁
锁 一般 。。 只升级 不降级
(2)复杂版
CAS 简单叙述 了解入门
什么是ABA问题,假如你有媳妇儿,我说假如~ 以偷零花钱为例
https://www.processon.com/view/link/603c96ca07912913b4f2c55f
这是一个故事: 小强偷零花钱请小明吃饭的故事
ABA 就是:
小强媳妇儿 出门看的钱是10万 (A)
小强偷拿1万 剩余9万(B)
小强找小月借了1万 放回去 总共10万(A)
小强媳妇儿回来 一看是10万,很满意。 但是 她不知道 这是偷梁换柱啊
后来小强媳妇儿看了我的博客 , 发现了秘密 ,她应该怎么解决呢 ?
关键字:version 版本号
她上班之前,在家里的存款上用笔写了一个版本,小强如果再偷钱,再还回来,这个版本就变了(+1)
这样就解决了ABA问题
2. 上锁过程
重度竞争: 耗时过长 自选过多 wait等
新建对象可能直接是匿名偏向 ( 如果默认开启了偏向锁) ,因为没有偏向任何一个线程,所以是匿名偏向
JVM默认不开启 延迟4秒后才会开启 偏向锁
3. new 一个对象 长什么样 ? markword 是啥
1. 查看工具 : JOL (Java Object Layout )
直接maven引入就可以使用了
org.openjdk.jol jol-core 0.9
/** * @author 木子的昼夜 */public class Person { long money; public long getMoney() { return money; } public void setMoney(long money) { this.money = money; }}/** * @author 木子的昼夜 */public class JolTest { public static void main(String[] args) { Person p = new Person(); System.out.println(ClassLayout.parseInstance(p).toPrintable()); }}// 输出结果 Person object internals: OFFSET SIZE TYPE DESCRIPTION VALUE 0 4 (object header) 01 00 00 00 4 4 (object header) 00 00 00 00 8 4 (object header) 43 c1 00 f8 12 4 (alignment/padding gap) 16 8 long Person.money 0Instance size: 24 bytesSpace losses: 4 bytes internal + 0 bytes external = 4 bytes total
咦? 不是要写synchronized 吗 ?
2. markword
我给对象p 加锁,然后输出 layout 可以发现 markword 改变了
所以呢~~ 锁信息 是记录再markword中的
/** * @author 木子的昼夜 */public class JolTest { public static void main(String[] args) { Person p = new Person(); System.out.println(ClassLayout.parseInstance(p).toPrintable()); synchronized (p) { System.out.println(ClassLayout.parseInstance(p).toPrintable()); } }}
markword 这么厉害吗? 不 ! 它还能更厉害 。 我们看一下 它里边都记录了一些什么信息
先暂时看最后3bit 其他 不是很了解 这个时候可以对比上边layout的输出 看一下
刚开始是01 加锁之后变成了00 (没有开启偏向锁 直接到轻量级锁)
先看最后2bit:
00:轻量级锁 自旋锁
自旋锁,耗CPU资源 是在用户态操作 不关联内核态
两个线程 争着把自己的Lock Record 放到markword中
谁先放进去,谁先获得锁,另一个人接着cas 去放
Lock Record 指向的是什么呢 是无锁状态的markword
这就解释了 为什么hashcode不丢失的问题 因为有备份记录
这里锁重入: 上边提到了,锁重入 ,锁每进一次,都会加一个LR 从第二个LR开始 指向的就是一个null
等锁退出 也就是monitorexit(锁代码块执行完 或 抛异常)的时候LR -1 ,LR -1 ,LR -1 一直减 ,退一次减一次
10:重量级锁
向OS 申请锁,进了内核态 , c++ 新建了一个object monitor对象 markword中放的就是这个 指针 (java中就是个地址或者是ID )
重量级锁,都在一个队列里等着,比较不消耗CPU资源
可重入锁: 重量级是记录再object moniter 的某个属性上
什么时候自选上升为重量级锁:
1. 自选次数超过10次 或者 自选的线程数超过CPU的一半 2. jdk1.6之前 -XX:PreBlockSpin可以调整 自选超过多少次升级 3. jdk1.6之后加入了自适应 Adapative Self Sping JVM自己个儿控制
11:GC回收标记
01: 再看倒数第三位
0:无锁
1:偏向锁 放线程ID , c++实现是用的指针
3. synchronized 编译成字节码 会有两个单词 monitorenter monitorexit
什么时候monitorexit呢 , 代码执行完 ,或者是异常发生 这就是synchronized 自动释放锁的原理
4. 为什么有自旋锁还需要重量级锁
(1) 自旋是消耗CPU资源的,如果锁的时间长,或者自旋线程多,CPU会被大量消耗
(2) 重量级锁有等待队列,所有拿不到锁的进入等待队列,不需要消耗CPU资源
5. 偏向锁是否一定比自旋锁效率高?
(1)不一定 当你知道肯定存在多线程竞争的时候,偏向锁会涉及锁撤销,这时候自旋锁会比较好一点
(2) JVM启动过程就会很很多线程竞争,所以默认不开启偏向锁,过一段时间才会开启
(3) -XX:BiasedLockingStartupDelay = 0 默认是 4 秒
感谢各位的阅读,以上就是"怎么使用synchronized "的内容了,经过本文的学习后,相信大家对怎么使用synchronized 这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是,小编将为大家推送更多相关知识点的文章,欢迎关注!