【Java并发编程的艺术】第二章读书笔记之synchronized关键字

在以前的文章中学习了volatile关键字,volatile能够保证变量在线程间的可见性,但他不能真正的保证线程安全。java

/** * @author cenkailun * @Date 9/5/17 * @Time 20:23 */
public class ConcurrentAddWithVolatile implements Runnable {

    private static ConcurrentAddWithVolatile instance = new ConcurrentAddWithVolatile();
    private static volatile int i = 0;


    public static void increase() {
        i++;
    }

    public void run() {
        for (int j = 0; j < 1000000; j++) {
            i++;
        }
    }

    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(instance,"线程1");
        Thread t2 = new Thread(instance, "线程2");
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println(i);
    }
}复制代码

如上述代码所示,若是说两个线程是正确的并发执行的话,最后获得的结果应该是2000000,但结果每每是小于2000000。那么这是为何呢?安全

通过阅读书籍,能够得知,i++的这个操做,实际上是要分红3步。bash

1. 读取i的当前值到操做栈
2. 对i的当前值+1
3. 写回i+1后的值复制代码

通过了上述3步,才完成了i++ 的这个操做,volatile保证了写回内存后,i的最新值可以被其余线程获取,但i++的这三个动做不是一个总体,即不是原子操做,是能够被拆开的。多线程

好比,线程1和2同时读取了i为0,并各自在本身的线程中计算获得i=1,前后写入这个i的值,致使虽然i++被执行了两次,可是实际i的值只增长了1。并发

若是要解决这个问题,就要保证多个线程在对i进行++ 这个操做时彻底同步,即i++的这三步是一块儿完成的,当线程1在写入时,其余线程不能读也不能写,由于在线程1写完以前,其余线程读到的确定是一个过时的数据。jvm

Java提供了synchronized来实现这个功能,保证多线程执行时候的同步,某一时刻只有一个线程能够对synchronized关键字保护起来的区域进行操做,相对于volatile来讲是比较重量级的。学习

Java的synchronized关键字具体表现有如下三种形式:spa

  1. 做用于实例方法,锁的是当前实例对象。
  2. 做用于静态方法,锁的是当前类。
  3. 做用于代码块,锁的是Synchronized里配置的对象。

下面是一个示例,将synchronized做用于一个给定对象instance,每当线程要进入被包裹的代码块,会请求instance的锁。若是有其余线程已经持有了这把锁,那么新到的线程就必须等待,这样保证了每次只有一个线程会执行i++操做。线程

/** * @author cenkailun * @Date 9/5/17 * @Time 20:23 */
public class ConcurrentAddWithVolatile implements Runnable {

    private static ConcurrentAddWithVolatile instance = new ConcurrentAddWithVolatile();
    private static volatile int i = 0;


    public static void increase() {
        i++;
    }

    public void run() {
        for (int j = 0; j < 1000000; j++) {
            synchronized (instance) {     //同步代码块
                i++;
            }
        }
    }

    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(instance,"线程1");
        Thread t2 = new Thread(instance, "线程2");
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println(i);
    }
}复制代码

对于java中的代码块同步,JVM是基于进入和退出Monitor对象来实现代码块同步的,将monitorenter指令插入到同步代码块的开始位置,monitorexit插入到方法结束处和异常处,每个对象都有一个monitor与之对应,当一个monitor被持有后,它将处于锁定状态。线程执行到monitorenter指令时,将会尝试获取对象所对应的monitor的全部权,即尝试得到对象的锁。code

以下面字节码所示,表明上文代码中的同步代码块。

13: monitorenter
14: getstatic     #2 // Field i:I
17: iconst_1
18: iadd
19: putstatic     #2 // Field i:I
22: aload_2
23: monitorexit复制代码

对于实例方法或者静态方法上加的synchronized关键字,在方法上会有一个标志位表明,以下面字节码所示。

public synchronized void increase();
flags: ACC_PUBLIC, ACC_SYNCHRONIZED复制代码

在我看来,synchronized相对于volatile的强大之处在于保证了线程安全性以及作到了线程同步,同时也能作到volatile提供的线程间可见性以及有序性。从可见性上来讲,线程经过持有锁的方式获取变量的最新值。从有序性上来讲,synchronized限制每次只有一个线程能够访问同步的代码,不管内部指令顺序如何被打乱,jvm会保证最终执行的结果老是同样,其余线程只能在得到锁后读取结果数据,不会读到中间值,因此有序性问题也获得了解决。

相关文章
相关标签/搜索