synchronized关键字的原理

  • synchronized的执行严格遵守java happens-before 规则,一个monitor exit指令之前必定要有一个monitor enter。
synchronized关键字的用法

synchronized可以用于对代码块或方法进行修饰,而不能够用于对class以及变量进行修饰。

  • 同步方法

public synchronized void sync() {

//…

}

  • 同步方法块

private final Object lock = new Object();

public void sync() {

synchronized(lock) {

//…

}

}

关于同步代码块和同步方法的区别之前写过一个关于这个对比,具体可以看这篇文章。java中的synchronized(同步代码块和同步方法的区别)

深入分析Synchronized关键字
线程堆栈分析

synchronized关键字提供了一种互斥机制,也就是说在同一时刻,只能有一个线程访问同步资源。看下面这段程序:

import java.util.concurrent.TimeUnit;

public class TestSync {

private final static Object lock = new Object();

public void accessResource() {

synchronized(lock) {

try {

TimeUnit.MINUTES.sleep(10);

} catch (InterruptedException e) {

e.printStackTrace();

}

}

}

public static void main(String[] args) {

final TestSync sync = new TestSync();

for(int i =0;i<5;i++) {

new Thread(){

@Override

public void run() {

sync.accessResource();

}

}.start();

}

}

}

上面的代码定义一个方法accessResource,并且使用synchronized来对代码进行同步,同时定义了5个线程调用accessResource方法,由于synchronized的互斥性,只能有一个线程获得lock的monitor锁,其他线程只能进入阻塞状态,等待获取lock的monitor锁。针对这个monitor锁我们如何从线程堆栈信息来看呢?其实,jstack命令在Java中可以用来打印进程的线程堆栈信息。我们来运行这个Java程序,在终端通过top命令查看运行起来的Java程序的进程id,然后执行jstack ‘pid’。我们来看下打印出来的信息:

通过截图可以看到Thread-0持有monitor<0x00000007955f2130>的锁并且处于休眠状态中,而其他几个线程则是处于BLOCKED状态中,它们是在等待着获取monitor<0x00000007955f2130>的锁。

JVM指令分析

从JVM指令角度再来分析synchronized关键字。我们可以使用javap这个命令来对上面这个TestSync类生成的class字节码进行反编译,得到下面的JVM指令。

Compiled from “TestSync.java”

public class main.TestSync {

static {};

Code:

0: new #3 // class java/lang/Object

3: dup

4: invokespecial #10 // Method java/lang/Object.“”😦)V

7: putstatic #13 // Field lock:Ljava/lang/Object;

10: return

public main.TestSync();

Code:

0: aload_0

1: invokespecial #10 // Method java/lang/Object.“”😦)V

4: return

public void accessResource();

Code:

0: getstatic #13 // Field lock:Ljava/lang/Object;

3: dup

4: astore_1

5: monitorenter

6: getstatic #20 // Field java/util/concurrent/TimeUnit.MINUTES:Ljava/util/concurrent/TimeUnit;

9: ldc2_w #26 // long 10l

12: invokevirtual #28 // Method java/util/concurrent/TimeUnit.sleep:(J)V

15: goto 23

18: astore_2

19: aload_2

20: invokevirtual #32 // Method java/lang/InterruptedException.printStackTrace:()V

23: aload_1

24: monitorexit

25: goto 31

28: aload_1

29: monitorexit

30: athrow

31: return

Exception table:

from to target type

6 15 18 Class java/lang/InterruptedException

6 25 28 any

28 30 28 any

public static void main(java.lang.String[]);

Code:

0: new #1 // class main/TestSync

3: dup

4: invokespecial #44 // Method “”😦)V

7: astore_1

8: iconst_0

9: istore_2

10: goto 27

13: new #45 // class main/TestSync$1

16: dup

17: aload_1

18: invokespecial #47 // Method main/TestSync$1.“”:(Lmain/TestSync;)V

21: invokevirtual #50 // Method main/TestSync$1.start:()V

24: iinc 2, 1

27: iload_2

28: iconst_5

29: if_icmplt 13

32: return

最后

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助

因此我收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
831877797)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

  • 3
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值