synchronized详解 (上)

简介: synchronized详解 (上)

一.设计同步器的意义



  多线程编程中,有可能会出现多个线程同时访问同一个共享可变资源的情况,这个资源我们称之其为临界资源;这种资源可能是:对象、变量、文件等。

 

  共享:资源可以由多个线程同时访问

 

  可变:资源可以在其生命周期内被修改

 

  引出的问题:

     由于线程执行的过程是不可控的,所以需要采用同步机制来协同对对象可变状态的访问!


1.1  如何解决线程并发安全问题?


实际上,所有的并发模式在解决线程安全问题时,采用的方案都是序列化访问临界资源。即在同一时刻,只能有一个线程访问临界资源,也称作同步互斥访问。

Java 中,提供了两种方式来实现同步互斥访问:synchronized 和 Lock

同步器的本质就是加锁


加锁目的:序列化访问临界资源,即同一时刻只能有一个线程访问临界资源(同步互斥访问)


不过有一点需要区别的是:当多个线程执行一个方法时,该方法内部的局部变量并不是临界资源,因为这些局部变量是在每个线程的私有栈中,因此不具有共享性,不会导致线程安全问题。


二. synchronized原理详解



synchronized内置锁是一种对象锁(锁的是对象而非引用),作用粒度是对象,可以用来实现对临界资源的同步互斥访问,是可重入的。


2.1 Synchronized的发展历史

1187916-20200907062312336-354442008.png

如上图Synchronized的发展经历了3个阶段. 如上图: 橙色第一阶段, 黄色第二阶段和绿色的第三阶段

第一阶段:

1. jdk<1.6版本的之前, synchronized的效率非常的低, 因为,当时synchronized依赖的加锁方式是java对象锁
2. 当创建java对象的时候, 也就是new Object()的时候,都会天然的创建一个管存对象Monitor
3. synchronized如何加锁成功呢?它依赖于管存对象, 而管存对象依赖于底层的操作系统OS里的Mutex互斥量
4. mutex互斥量是由底层的操作系统维护的, mutex执行的时候会去调用底层的线程库,在linux里面统称为Pthread线程库,
设计到阻塞、升级互斥量等操作,这个是依赖于操作系统的。
之前说过, 我们的底层分为用户空间和内核空间, jvm是在用户空间, 如果他要去调用底层的内核空间是很耗时的。 
所以,刚开始synchronized的效率是非常低的

第二阶段:


这个效率实在是太低了, 于是,Dong li就实现了一个AQS并发框架,里面使用java语言实现了ReentrantLock,  

ReentrantLock实现可重用锁和所有的公性


第三阶段:


后来sun公司被Oracle收购了以后, Synchronized是java的亲儿子, 现在他的性能缺这么低, 还不如ReentrantLock,这怎么可以呢?
于是oracle就优化了Synchnized,在jdk
>= 1.6版本以后, 使用的就是升级后的Synchronized锁了. 用户最开始使用的是对象的偏向锁,当并发较大的时候,切换到java轻量级锁,


当并发量很大的时候, 才切换到操作操作系统底层的Mutex互斥量锁


什么是偏向锁?


刚开始的时候是无锁的状态, 所以,最开始只有一个线程, 没有竞争, 此时的java对象是没有锁的,所以,使用的就是偏向锁。也就是单线程的情况下使用的是偏向锁


什么是轻量级锁?


上面是只有一个线程的情况, 那么如果有两个线程同时调用一个程序,那么他们之间就会存在竞争, 但是竞争又不是很激烈, 就升级为轻量级锁

如何实现轻量级锁呢?在锁的内部增加一个while循环, 比如加入while循环100次, 如果100次循环结束以后, 锁还没有释放, 那么就会升级为重量级锁。 如果释放了, 那么说明等待的时间不是很久,


就不需要升级。 升级的时候也是在内部直接升级


何时升级为重量级锁


当竞争更加激烈的时候, 轻量级锁在指定的循环内还没有释放锁, 就说明竞争很激烈了, 这时,就将其升级为重量级锁。


jdk1.6以后, 锁从偏向锁升级到轻量级锁,再到重量级锁, 这个过程是否是可逆的呢?

答案是,这个过程是不可逆的。 因为从重量级锁在退回到轻量级锁也是没有意义的。 为什么呢?你的并发已经很多的。 再退到轻量级锁, 然后再次进行升级, 这样没有意义。

而且锁的升级也是对效率有影响


2.2 加锁的方式:


1、同步实例方法,锁是当前实例对象

public class LockOnClass {
    static int stock;
    public static synchronized void decrStock() {
        System.out.println(--stock);
    }
    public synchronized void addStock() {
        System.out.println(++ stock);
    }
    public static void main(String[] args) {
        LockOnClass.decrStock();
    }
}

这种类型的锁是加载方法级别上的, 加在方法级别的synchronized, 是类实例界别的锁.

 

2、同步类方法,锁是当前类对象

package com.company;
public class LockOnClass {
    static int stock;
    public static synchronized void decrStock() {
        System.out.println(--stock);
    }
    public static void main(String[] args) {
        LockOnClass.decrStock();
    }
}

可以看到, 当前的锁是加载静态方法上的, 静态方法是类打点直接调用. 也就是说, 这类同步是类级别的同步, 他们的锁是类名.

 

重点说明 ,看下面这个例子

package com.company;
public class LockOnClass {
    static int stock;
    public static synchronized void decrStock() {
        System.out.println(--stock);
    }
    public static synchronized void addStock() {
        System.out.println(++ stock);
    }
    public static void main(String[] args) {
        LockOnClass.decrStock();
    }
}

这是在一个类的两个静态方法上加了synchronized, 这是非常非常需要重点注意的问题.

在静态方法上家synchronized, 他是类级别的加锁. 两个方法同时加锁, 会大大降低程序的性能.


如果这两个方法被到处调用, 这将是一个灾难性的问题. 这对系统的qps将有一个很大的影响.

 

重点说明2: 为什么不能再项目里大量的写

System.out.println("");


System.out.println(--stock);

1187916-20200907155125791-1160767167.png

原因在这里, out是System类中的一个静态变量. 也就是全局只有一个

再来看看println方法

1187916-20200907155222627-848042908.png

我们看到println里面有一个synchronized同步标志.

而且,其他的类似代码, 如下图

1187916-20200907155315771-404342287.png

我们发现他们都有synchronized关键字修饰. 这意味着什么呢?


他们的调用对象都是System.out. 也就是锁是全局唯一的out对象. 如果代码里有很多很多System.out.println(), 那么将大大降低程序的性能.


原因是他们持有的都是一把锁, 要进行锁等待. 所以, 代码里不可以有System.out.println(); 会严重影响我们的qps和tps.

 

3、同步代码块,锁是括号里面的对象

public class LockOnObject {
    public static Object object = new Object();
    private Integer stock = 10;
    public void decrStock() {
        synchronized (object) {
            -- stock;
            if (stock <= 0) {
                System.out.println("库存售罄");
                return;
            }
        }
    }
}

这是对一块代码块进行同步. 锁是定义的一个全局的objct对象.

 

2.2 synchronized底层原理


2.1 字节码层面的锁标志


jvm怎么知道加了synchronized就要加锁呢? 其实, 是在字节码层面加了特殊标志.


1. 同步代码块的字节码

public void decrStock() {
        synchronized (object) {
            -- stock;
            if (stock <= 0) {
                System.out.println("库存售罄");
                return;
            }
        }
    }

我们来看看这段代码的字节码文件, 被翻译成字节码的时候, 会被翻译成特殊的指令

1187916-20200907163114315-1369533606.png


加锁的时候, 增加了monitorenter关键字

1187916-20200907163210480-98109295.png

然后在后面会有monitorexit, 进行退出.

为什么会有三个monitorexit呢? 这是因为进行了异常情况的捕获处理.

2. 同步类方法的字节代码

public static synchronized void decrStock() {
        System.out.println(--stock);
}

我们来看看被synchronized修饰的静态方法的字节码

1187916-20200907163626203-2137137738.png

我们看到, 在静态方法里面增加了ACC_SYNCHRONIZED标志. 这个标志有点类似与volatile的特殊标志. 有这个标志的代码, 及表示是同步代码块


当jvm执行代码的时候, 遇到ACC_SYNCHRONIZED标识符, 在底层会自动加上monitorenter和monitorexit.


3. Monitor监视器锁


任何一个对象都有一个Monitor与之关联,当且一个Monitor被持有后,它将处于锁定状态。


Synchronized在JVM里的实现都是基于进入和退出Monitor对象来实现方法同步和代码块同步,虽然具体实现细节不一样,但是都可以通过成对的MonitorEnter和MonitorExit指令来实现。


  • monitorenter:每个对象都是一个监视器锁(monitor)。当monitor被占用时就会处于锁定状态,线程执行monitorenter指令时尝试获取monitor的所有权,过程如下:
  1. 如果monitor的进入数为0,则该线程进入monitor,然后将进入数设置为1,该线程即为monitor的所有者;
  2. 如果线程已经占有该monitor,只是重新进入,则进入monitor的进入数加1;
  3. 如果其他线程已经占用了monitor,则该线程进入阻塞状态,直到monitor的进入数为0,再重新尝试获取monitor的所有权;
  • monitorexit:执行monitorexit的线程必须是objectref所对应的monitor的所有者。指令执行时,monitor的进入数减1,如果减1后进入数为0,那线程退出monitor,不再是这个monitor的所有者。其他被这个monitor阻塞的线程可以尝试去获取这个 monitor 的所有权。


monitorexit,指令出现了两次,第1次为同步正常退出释放锁;第2次为发生异步退出释放锁;


  通过上面两段描述,我们应该能很清楚的看出Synchronized的实现原理,Synchronized的语义底层是通过一个monitor的对象来完成其实wait/notify等方法依赖于monitor对象,这就是为什么只有在同步的块或者方法中才能调用wait/notify等方法,否则会抛出java.lang.IllegalMonitorStateException的异常的原因。


看一个同步方法:

package it.yg.juc.sync;
public class SynchronizedMethod {
    public synchronized void method() {
        System.out.println("Hello World!");
    }
}

反编译结果:

1187916-20200907163626203-2137137738.png


从编译的结果来看,方法的同步并没有通过指令 monitorenter 和 monitorexit来完成(理论上其实也可以通过这两条指令来实现),不过相对于普通方法,其常量池中多了 ACC_SYNCHRONIZED 标示符JVM就是根据该标示符来实现方法的同步的:

 

  当方法调用时,调用指令将会检查方法的 ACC_SYNCHRONIZED 访问标志是否被设置,如果设置了,执行线程将先获取monitor,获取成功之后才能执行方法体,方法执行完后再释放monitor。


  在方法执行期间,其他任何线程都无法再获得同一个monitor对象。


  两种同步方式本质上没有区别,只是方法的同步是一种隐式的方式来实现,无需通过字节码来完成。


  两个指令的执行是JVM通过调用操作系统的互斥原语mutex来实现,被阻塞的线程会被挂起、等待重新调度,会导致“用户态和内核态”两个态之间来回切换,对性能有较大影响。

相关文章
|
4月前
|
算法 Java 编译器
Synchronized你又知道多少?
Synchronized 是 JVM 实现的一种互斥同步机制,通过 monitorenter 和 monitorexit 指令控制对象锁的获取与释放。锁的本质是对象头的标记,确保同一时间只有一个线程访问资源。Synchronized 支持可重入性,允许方法内部调用其他同步方法而不阻塞。JVM 对锁进行了优化,引入了自旋锁、偏向锁、轻量级锁和重量级锁,以减少系统开销。Synchronized 属于悲观锁,而乐观锁基于 CAS(Compare and Swap)算法实现非阻塞同步,提高并发性能。
77 6
|
8月前
|
Java
synchronized
synchronized
43 2
|
8月前
|
存储 安全 Java
|
8月前
synchronized与ReentrantLock区别与联系
synchronized与ReentrantLock区别与联系
44 0
|
安全 算法 Java
synchronized 同步锁
Java中的synchronized关键字用于实现线程同步,可以修饰方法或代码块。 1. 修饰方法:当一个方法被synchronized修饰时,只有获得该方法的锁的线程才能执行该方法。其他线程需要等待锁的释放才能执行该方法。 2. 修饰代码块:当某个对象被synchronized修饰时,任何线程在执行该对象中被synchronized修饰的代码块时,必须先获得该对象的锁。其他线程需要等待锁的释放才能执行同步代码块。Java中的每个对象都有一个内置锁,当一个对象被synchronized修饰时,它的内置锁就起作用了。只有获得该锁的线程才能访问被synchronized修饰的代码段。使用synch
69 0
ReentrantLock和Synchronized简单比较
ReentrantLock和Synchronized简单比较
52 0
|
Java
07.synchronized都问啥?
大家好,我是王有志。经过JMM和锁的铺垫,今天我们正式进入synchronized的内容,来看看关于synchronized面试中都会问啥?
68 1
07.synchronized都问啥?
Synchronized
作用:能够保证在同一时刻最多有一个线程执行该段代码,以保证并发的安全性。(当第一个线程去执行该段代码的时候就拿到锁,并独占这把锁,当方法执行结束或者一定条件后它才释放这把锁,在没释放锁之前,所有的线程处于等待状态)
75 0
synchronized的总结
synchronized的总结
101 0