一、Synchronized的概念

是利用锁的机制来实现同步的。

锁机制有如下两种特性:

互斥性:即在同一时间只允许一个线程持有某个对象锁,通过这种特性来实现多线程中的协调机制,这样在同一时间只有一个线程对需同步的代码块(复合操作)进行访问。互斥性我们也往往称为操作的原子性。

可见性:必须确保在锁被释放之前,对共享变量所做的修改,对于随后获得该锁的另一个线程是可见的(即在获得锁时应获得最新共享变量的值),否则另一个线程可能是在本地缓存的某个副本上继续操作从而引起不一致。

二、Synchronized的使用

修饰静态方法

  

//修饰静态方法
public static synchronized void print() {
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"runing");
}

  

修饰非静态方法

  

//修饰非静态方法
public synchronized void print1() {
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"runing");
}

  

修饰代码块

//代码块的使用
public void prnit2() {
synchronized (this) {
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"runing");
} } //代码块的使用
public void prnit3() {
synchronized (SynchroTest.class) {
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"runing");
} }

  

疑问:修饰代码块的时候,Synchronized参数对象起到了什么作用?

答:当JVM是用ClassLoader加载字节码的时候,会在方法区创建一个对象,同时也会在堆去创建一个Class(注意是大写的)。使用Synchronized.xxx.class就是说明所有Class对应的对象都使用共同一个锁(比较抽象)。详情请看原理分析JVM指令分析的 Monitorenter,Monitorexit。JVM会将对象与Monitor相关联。

java中每一个对象都会有一个monitor对象(监视器)。它的作用就是给对象加锁用的。

  1. 某一线程占有这个对象的时候,先看monitor的计数器是不是为0,如果是0说明还没有线程占有,此时会将monitor的计数器+1.如果不为0,表示有其他线程占有这个对象,需要等待。当占有这个对象的线程释放这个对象的时候,那么此时会将这个对象的monitor-1(并不是monitor=0,而是-1)。类似于CPU中的cache line作用。

三、Synchronized的原理分析

线程堆栈分析(互斥性)

public static void main(String[] args) throws InterruptedException {
final SynchroTest tSynchroTest1 = new SynchroTest();
for (int i = 10; i < 15; i++) {
new Thread(tSynchroTest1::print1).start();
}
} // 修饰非静态方法
public synchronized void print1() {
try {
TimeUnit.MINUTES.sleep(2);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + "runing");
}

  

根据以上代码进行分析。

    • jconsole分析,可以看出当前拥有对象锁的Thread-3因为线程睡眠,所以状态为TIMED_WAITING(等待状态)。看Thread-2可以看出总阻止数为1,也就是BLOCKED状态(上锁状态)。

    • jstack pid分析也可以看到dos命令窗口中的当前用对象线程是TIME_WAITING,阻塞线程是BLOCKED

  • JVM的指令分析
  1. javap(反编译)使用cmd,找到calss文件路径  D:\workspace1\GuaHao\bin\BlockQueueTest>javap -v SynchroTest。代码块的一个加锁方式,是使用Monitorenter、Monitorexit配合使用

    • 细心的朋友上图会看到51有一个monitorexit  和 56还有一个monitorexit?为什么会有两个互斥出口
    • 答:51行是正常出口,56是一个异常出口。当程序发生运行时异常的时候会从56monitorexit跳出。

    2.方法加锁方式,是使用、配合使用

  对象与monitor:一个实例对象包含了对象头,实例变量,填充数据。

  对象头:加锁的一个基础。

  实例变量:私有变量,属性变量信息

  填充数据:对象的启示地址,8字节表示。

对象头

  

  • JDK6之前 Synchronized是重量锁,比较暴力。
  • JDK6之后 添加了

    1. 偏向锁:再对象第一次被某一线程占有的时候,是否偏向锁置1,锁表01,写入线程号;当其他的线程访问的时候,就会竞争,结果分别是失败和成功。很多次被第一线程占有它的线程获得次数比较多此时就是偏向锁。相当于找对象(女朋友),当这个对象与第一个男人分手后,找到第二个男人之后,可能还会去找前男友,这就是属于偏向锁。竞争不激烈的时候使用。偏向锁与无所状态时间很接近。CAS算法campare and set(留个小作业,什么叫CAS算法?)
    2. 轻量级锁:线程有交替适用,互斥性不是很强,使用CAS算法不在适用,锁标志00。
    3. 重量级锁:强互斥,锁标记10,等待时间长。

      他们的区别是时间上的区别。

      用户线程和核心线程相互转换(非常耗时)

      由偏向锁转轻量级锁在转重量级锁的时候存在不可逆行为,只能低转高,不可反之。

      自旋锁:再转换过程中还存在一个叫做自旋锁的情况,再轻量级装重量级的时候,并不是马上从用户线程转成核心线程,再互斥的情况下会执行几 次空循环,如果空循环完成后再没有拿到占有权,则才会进行转换。

      消除锁:JIT(编译)的时候如果定义一个int i=0,并对这个定义进行加锁,那么由于上篇文章说道了,定义int i=0,是存在原子性的,所以这个这 个锁会在编译期间自动消除,防止JVM产生不必要的资源浪费

      CAS算法:引用GA特大的Java VAS原理深度分析文章  https://blog.csdn.net/tanga842428/article/details/52742698

      

欢迎大家来提问,如果有不理解的可以再评论区提出,我会定时解惑。

同舟共济,新海同航

什么是偏向锁:如何证明?
跟着路神一起开车证明呀,哈哈。我们的Synchronized关键字会调用OS(操作系统)中的pthread_mutex_lock()方法,然后修改此方法,再此方法中打印一句话,如果每次都输出说明次Synchronized非偏向锁。如果只打印一次,说明我们的synchronized本次是偏向锁,如此就证明了JVM中的偏向锁的存在。Synchronized关键字会再第一次调pthread_mutex_lock(),如果是偏向锁,则不在调用,非偏向锁则会多次调用

     

并发2-Synchronized的更多相关文章

  1. 5.并发编程-synchronized 细节说明

    并发编程-synchronized 细节说明 1. synchronized-锁重入 & 异常释放锁 说明 * 关键字synchronized 拥有锁重入的功能,也就是在使用synchroni ...

  2. Java并发分析—synchronized

    在计算机操作系统中,并发在宏观上是指在同一时间段内,同时有多道程序在运行. 一个程序可以对应一个进程或多个进程,进程有独立的存储空间.一个进程包含一个或多个线程.线程堆空间是共享的,栈空间是私有的.同 ...

  3. Java并发编程-synchronized

    多线程的同步机制对资源进行加锁,使得在同一个时间,只有一个线程可以进行操作,同步用以解决多个线程同时访问时可能出现的问题.同步机制可以使用synchronized关键字实现.synchronized关 ...

  4. 悲观的并发策略——Synchronized互斥锁

    volatile既然不足以保证数据同步,那么就必须要引入锁来确保.互斥锁是最常见的同步手段,在并发过程中,当多条线程对同一个共享数据竞争时,它保证共享数据同一时刻只能被一条线程使用,其他线程只有等到锁 ...

  5. Java并发编程-synchronized指南

    在多线程程序中,同步修饰符用来控制对临界区代码的访问.其中一种方式是用synchronized关键字来保证代码的线程安全性.在Java中,synchronized修饰的代码块或方法不会被多个线程并发访 ...

  6. Java并发—–深入分析synchronized的实现原理

    记得刚刚开始学习Java的时候,一遇到多线程情况就是synchronized,相对于当时的我们来说synchronized是这么的神奇而又强大,那个时候我们赋予它一个名字“同步”,也成为了我们解决多线 ...

  7. 对象和变量的并发访问synchronized解析以及死锁分析排查

    一.synchronized java并发编程中存在“非线程安全"问题.“非线程安全"是指发生在多个线程对同一个对象中的实例变量并发访问时,产生的”脏读“现象,使用synchron ...

  8. 多线程与高并发(三)synchronized关键字

    上一篇中学习了线程安全相关的知识,知道了线程安全问题主要来自JMM的设计,集中在主内存和线程的工作内存而导致的内存可见性问题,及重排序导致的问题.上一篇也提到共享数据会出现可见性和竞争现象,如果多线程 ...

  9. Java并发,synchronized锁住的内容

    synchronized用在方法上锁住的是什么? 锁住的是当前对象的当前方法,会使得其他线程访问该对象的synchronized方法或者代码块阻塞,但并不会阻塞非synchronized方法. 脏读 ...

  10. java并发编程--Synchronized的理解

    synchronized实现锁的基础:Java中每一个对象都可以作为锁,具体表现为3种形式. (1)普通同步方法,锁是当前实例对象 (2)静态同步方法,锁是当前类的Class对象 (3)同步方法块,锁 ...

随机推荐

  1. Java语法基础练习2

    ---恢复内容开始--- 1.仔细阅读示例:EnumTest.java分析结果 代码: 运行结果: 分析:枚举类型就是一个类,枚举中的常量就是枚举类型中的实例,可把字符串转化为枚举:而且他本身是一个类 ...

  2. JAVA JVM 流程一

    JVM是Java Virtual Machine(Java虚拟机)的缩写,JVM是一种用于计算设备的规范,它是一个虚构出来的计算机,是通过在实际的计算机上仿真模拟各种计算机功能来实现的.Java虚拟机 ...

  3. STM32:获取复位源,软件复位

    RCC CSR寄存器会存储复位标示,可通过它来知道复位原因,来源: if(RCC_GetFlagStatus(RCC_FLAG_PINRST)) printf("PINRST\r\n&quo ...

  4. 解决Linux主机上的 远程MySQL客户端无法连接的问题

    无法连接到 MySQL 数据库可能的原因有: 1. PHP 无法连接 MySQL 可能是 PHP 配置不正确,没加上连接 MySQL 的功能. 2. MySQL 软件包升级,但没有升级数据库,或安装 ...

  5. UVa 1161 Objective: Berlin (最大流)

    题意:给定一些航班,每个航班有人数,和起始终止时间,每次转机要花半小时,问限制时间内最多能有多少人从起始城市到终点城市. 析:差不多是裸板网络流的最大流问题,把每个航班都拆成两个点,这两个点之间连接一 ...

  6. bzoj 2229: [Zjoi2011]最小割【Gomory–Hu tree最小割树】

    这个算法详见http://www.cnblogs.com/lokiii/p/8191573.html 求出两两之间最小割之后暴力统计即可 #include<iostream> #inclu ...

  7. 【爬坑系列】之解读kubernetes的认证原理&实践

    对于访问kube-apiserver模块的请求来说,如果是使用http协议,则会顺利进入模块内部得到自己想要的:但是如果是用的是https,则能否进入模块内部获得想要的资源,他会首先要进行https自 ...

  8. 通过IDEA制作包含Java应程序的Docker镜像

    IDEA官网在IDEA中把Java App制作成Docker镜像并启动一个容器运行 在idea上使用docker作为java的开发环境[][] ubuntu+docker+docker-compose ...

  9. [ZOJ1610]Count the Colors

    Description 画一些颜色段在一行上,一些较早的颜色就会被后来的颜色覆盖了. 你的任务就是要数出你随后能看到的不同颜色的段的数目. Input 每组测试数据第一行只有一个整数n, 1 < ...

  10. 题解报告:hdu1995汉诺塔V(递推dp)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1995 Problem Description 用1,2,...,n表示n个盘子,称为1号盘,2号盘,. ...