在上一次https://www.cnblogs.com/webor2006/p/11428408.html中对于synchronized关键字的作用做了一个实例详解,下面再来看一下这个程序:

请问下,如果一个线程访问了同一个对象的method1()方法之后,另外一个线程能否访问同一个对角的method4的静态方法呢?答案是肯定的,因为method1的锁是锁的对象,而method4锁的是MyClass的类对象,锁是不一样的,当然就可以并行的进行访问啦,

关于synchronized其实从使用角度来说是比较容易理解的,但是要想充分理解它的底层其实并不是那么简单的,“偏向锁”、“轻量级锁”、“重量级锁”、“自旋锁”,可能提到这些东东顺间就懵逼了,这些其实都是在底层所能隐射出来的知识点,所以接下来会多底层来审视这个synchronized,下面一点点来,先看一下新的例子:

对于这个代码其实在实际代码中是非常之常见的,那提个疑问:

其实,改成任何对象其效果都是一模一样的,比如改一下:

而为啥都用Object可能是成了一种标准了,所以实际代码中同步块中都会去锁一个Object对象,这里要注意:其实声明任何类型的对象其效果都是一模一样的,至于为啥在未来的底层探索中会来揭示的。

下面对这个简单的程序进行一下字节码的反编译来看一下synchronized在底层的表现:

xiongweideMacBook-Pro:main xiongwei$ javap -c com/javacurrency/test3/MyTest1.class
Compiled from "MyTest1.java"
public class com.javacurrency.test3.MyTest1 {
public com.javacurrency.test3.MyTest1();
Code:
0: aload_0
1: invokespecial #1 // Method java/lang/Object."<init>":()V
4: aload_0
5: new #2 // class java/util/Date
8: dup
9: invokespecial #3 // Method java/util/Date."<init>":()V
12: putfield #4 // Field object:Ljava/util/Date;
15: return public void method();
Code:
0: aload_0
1: getfield #4 // Field object:Ljava/util/Date;
4: dup
5: astore_1
6: monitorenter
7: getstatic #5 // Field java/lang/System.out:Ljava/io/PrintStream;
10: ldc #6 // String hello world
12: invokevirtual #7 // Method java/io/PrintStream.println:(Ljava/lang/String;)V
15: aload_1
16: monitorexit
17: goto 25
20: astore_2
21: aload_1
22: monitorexit
23: aload_2
24: athrow
25: return
Exception table:
from to target type
7 17 20 any
20 23 20 any
}

其中如果想要看到更多的信息,比如常量池【这个在当时的JVM学习中已经详细学过了,就不过多解释了】,则可以用javap -v,如下:

xiongweideMacBook-Pro:main xiongwei$ javap -v com/javacurrency/test3/MyTest1.class
Classfile /Users/xiongwei/Documents/workspace/IntelliJSpace/java_javacurrency/build/classes/java/main/com/javacurrency/test3/MyTest1.class
Last modified 2019-8-30; size 719 bytes
MD5 checksum 44fd4afb0a37765559faa8ec0abb158f
Compiled from "MyTest1.java"
public class com.javacurrency.test3.MyTest1
minor version: 0
major version: 52
flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
#1 = Methodref #9.#26 // java/lang/Object."<init>":()V
#2 = Class #27 // java/util/Date
#3 = Methodref #2.#26 // java/util/Date."<init>":()V
#4 = Fieldref #8.#28 // com/javacurrency/test3/MyTest1.object:Ljava/util/Date;
#5 = Fieldref #29.#30 // java/lang/System.out:Ljava/io/PrintStream;
#6 = String #31 // hello world
#7 = Methodref #32.#33 // java/io/PrintStream.println:(Ljava/lang/String;)V
#8 = Class #34 // com/javacurrency/test3/MyTest1
#9 = Class #35 // java/lang/Object
#10 = Utf8 object
#11 = Utf8 Ljava/util/Date;
#12 = Utf8 <init>
#13 = Utf8 ()V
#14 = Utf8 Code
#15 = Utf8 LineNumberTable
#16 = Utf8 LocalVariableTable
#17 = Utf8 this
#18 = Utf8 Lcom/javacurrency/test3/MyTest1;
#19 = Utf8 method
#20 = Utf8 StackMapTable
#21 = Class #34 // com/javacurrency/test3/MyTest1
#22 = Class #35 // java/lang/Object
#23 = Class #36 // java/lang/Throwable
#24 = Utf8 SourceFile
#25 = Utf8 MyTest1.java
#26 = NameAndType #12:#13 // "<init>":()V
#27 = Utf8 java/util/Date
#28 = NameAndType #10:#11 // object:Ljava/util/Date;
#29 = Class #37 // java/lang/System
#30 = NameAndType #38:#39 // out:Ljava/io/PrintStream;
#31 = Utf8 hello world
#32 = Class #40 // java/io/PrintStream
#33 = NameAndType #41:#42 // println:(Ljava/lang/String;)V
#34 = Utf8 com/javacurrency/test3/MyTest1
#35 = Utf8 java/lang/Object
#36 = Utf8 java/lang/Throwable
#37 = Utf8 java/lang/System
#38 = Utf8 out
#39 = Utf8 Ljava/io/PrintStream;
#40 = Utf8 java/io/PrintStream
#41 = Utf8 println
#42 = Utf8 (Ljava/lang/String;)V
{
public com.javacurrency.test3.MyTest1();
descriptor: ()V
flags: ACC_PUBLIC
Code:
stack=3, locals=1, args_size=1
0: aload_0
1: invokespecial #1 // Method java/lang/Object."<init>":()V
4: aload_0
5: new #2 // class java/util/Date
8: dup
9: invokespecial #3 // Method java/util/Date."<init>":()V
12: putfield #4 // Field object:Ljava/util/Date;
15: return
LineNumberTable:
line 5: 0
line 6: 4
LocalVariableTable:
Start Length Slot Name Signature
0 16 0 this Lcom/javacurrency/test3/MyTest1; public void method();
descriptor: ()V
flags: ACC_PUBLIC
Code:
stack=2, locals=3, args_size=1
0: aload_0
1: getfield #4 // Field object:Ljava/util/Date;
4: dup
5: astore_1
6: monitorenter
7: getstatic #5 // Field java/lang/System.out:Ljava/io/PrintStream;
10: ldc #6 // String hello world
12: invokevirtual #7 // Method java/io/PrintStream.println:(Ljava/lang/String;)V
15: aload_1
16: monitorexit
17: goto 25
20: astore_2
21: aload_1
22: monitorexit
23: aload_2
24: athrow
25: return
Exception table:
from to target type
7 17 20 any
20 23 20 any
LineNumberTable:
line 9: 0
line 10: 7
line 11: 15
line 12: 25
LocalVariableTable:
Start Length Slot Name Signature
0 26 0 this Lcom/javacurrency/test3/MyTest1;
StackMapTable: number_of_entries = 2
frame_type = 255 /* full_frame */
offset_delta = 20
locals = [ class com/javacurrency/test3/MyTest1, class java/lang/Object ]
stack = [ class java/lang/Throwable ]
frame_type = 250 /* chop */
offset_delta = 4
}
SourceFile: "MyTest1.java"

这里重点来看一下方法,首先是默认构造方法:

当然它不是我们要看的重点,要看的是method()方法:

关于上面的字节码的所有信息都在当时JVM的学习中学习过了,也不过多说,只看跟同步相关的东东:

因为要锁对象首先肯定得先获取对象才行,如下:

接着就执行同步块里面的方法了:

也就是:

当同步块执行完了,则会看到字节码中会退出锁:

但是!!!

下次再说~

Java精通并发-透过字节码理解synchronized关键字的更多相关文章

  1. Java精通并发-同步方法访问标志与synchronized关键字之间的关系

    继续基于上一次https://www.cnblogs.com/webor2006/p/11428811.html来研究synchronized关键字在字节码中的表现,在上一次文末提出了一个这样的问题: ...

  2. Java精通并发-透过openjdk源码分析wait与notify方法的本地实现

    上一次https://www.cnblogs.com/webor2006/p/11442551.html中通过openjdk从c++的底层来审视了ObjectMonitor的底层实现,这次继续来探究底 ...

  3. 学以致用,通过字节码理解:Java的内部类与外部类之私有域访问

    目录: 内部类的定义及用处 打开字节码理解内部类 一.内部类的定义及用处 内部类(inner class)是定义在另一个类中的类.使用内部类,我们可以: 访问该类定义所在的作用域中的数据,包括私有的数 ...

  4. 深入理解java:1.2. 字节码执行引擎

    执行引擎是Java虚拟机的核心组成部分之一. 首先,想想C++和Java在编译和运行时到底有啥不一样? 下图左边,C++发布的就是机器指令, 而下图右边Java发布的是字节码,字节码在运行时通过JVM ...

  5. java之结合代码理解synchronized关键字

    为了保证数据的一致性即实现线程的安全性,java虚拟机提供了同步和锁机制.synchronized关键字是最基本的互斥同步手段.除此之外,还可以使用java.util.concurrent包中的重入锁 ...

  6. 并发编程学习笔记(3)----synchronized关键字以及单例模式与线程安全问题

    再说synchronized关键字之前,我们首先先小小的了解一个概念-内置锁. 什么是内置锁? 在java中,每个java对象都可以用作synchronized关键字的锁,这些锁就被称为内置锁,每个对 ...

  7. 【Java_多线程并发编程】基础篇——synchronized关键字

    1. synchronized同步锁的原理 当我们调用某对象的synchronized方法或代码块时,就获取了该对象的同步锁.例如,synchronized(obj)就获取了“obj这个对象”的同步锁 ...

  8. Java精通并发-wait与sleep方法字节码分析

    在上一次https://www.cnblogs.com/webor2006/p/11372521.html中对于Thread类和Runnable接口有了一个基本的认识,这次咱们继续巩固基础,首先先新建 ...

  9. 透过字节码分析java基本类型数组的内存分配方式。

    我们知道java中new方式创建的对象都是在堆中创建的,而局部变量对应的值存放在栈上.那么java中的int [] arr={1,2,3}是存放在什么地方的呢,int []arr = new int[ ...

随机推荐

  1. zabbix详解

    官网地址 https://www.zabbix.com/documentation/3.0/manual/config/items/itemtypes/zabbix_agent 使用率

  2. 【C/C++开发】C++实现字符串替换的两种方法

    替换字符串replace() erase() //C++ 第一种替换字符串的方法用replace()|C++ 第二种替换字符串的方法用erase()和insert()[ C++string|C++ r ...

  3. python基础篇(六)

    PYTHON基础篇(六) 正则模块re A:正则表达式和re模块案例 B:re模块的内置方法 时间模块time A:时间模块的三种表示方式 B:时间模块的相互转换 随机数模块random A:随机数模 ...

  4. 07 Mybatis的多表查询1----1对多和多对1---@Results注解用法总结

    1.表与表之间的关系及其举例 表之间的关系有4种:一对多.多对一.一对一.多对多. 举例: (1)用户和订单就是一对多 一个用户可以下多个订单 (2)订单和用户就是多对一 多个订单属于同一个用户 (3 ...

  5. 【翻译】在GitHub上通过星级评估排名前10的最受欢迎的开源Delphi项目

    GitHub上有相当多的Delphi开源项目可以为你节省一些时间.我在GitHub上搜索了Delphi,然后按最主要的项目进行排序,并列出了前十名单.这里有一些非常好的东西,包括Awesome Del ...

  6. pymysql 模块简单使用

    目录 pymysql 模块简单使用 安装 pymysql 模块 使用 pymysql 连接数据库 并插入数据 使用pymysql 插入数据 修改查询显示结果 pymysql 模块简单使用 安装 pym ...

  7. GC收集器

    新生代收集器 Serial New 单线程收集器,工作时必须暂停其他线程: 简单高效,没有线程交互开销: 基于复制算法: Parallel New 对Serial的改进,多线程: CPU数量<4 ...

  8. NOI2017

    整数(线段树) 不难想到按位处理,位数比较多考虑使用动态开点线段树维护大数,那么复杂度是\(O(nlog^2n)\)的,不够优秀. 但注意到我们需要支持的是二进制下的加减法,而在二进制下我们可以使用i ...

  9. NEST explain

    Elasticsearch 的相似度算法 被定义为检索词频率/反向文档频率, TF/IDF ,包括以下内容: 检索词频率 检索词在该字段出现的频率?出现频率越高,相关性也越高. 字段中出现过 5 次要 ...

  10. 【洛谷 P2444】 [POI2000]病毒(AC自动机)

    题目链接 这么多字符串,肯定是自动机啦. 先建出AC自动机,然后怎么表示一个安全代码没有病毒代码呢? 就是存在一条路径不经过有病毒代码段结尾的节点呗. 所以呢?有环啊!dfs一下救星了. #inclu ...