资讯专栏INFORMATION COLUMN

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

GT / 1027人阅读

摘要:在之前的文章中学习了关键字,可以保证变量在线程间的可见性,但他不能真正的保证线程安全。线程执行到指令时,将会尝试获取对象所对应的的所有权,即尝试获得对象的锁。从可见性上来说,线程通过持有锁的方式获取变量的最新值。

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

/**
 * @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步。

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写完之前,其他线程读到的肯定是一个过期的数据。

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

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

作用于实例方法,锁的是当前实例对象。

作用于静态方法,锁的是当前类。

作用于代码块,锁的是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的所有权,即尝试获得对象的锁。

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

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会保证最终执行的结果总是一样,其他线程只能在获得锁后读取结果数据,不会读到中间值,所以有序性问题也得到了解决。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/70341.html

相关文章

  • Java并发编程艺术二章读书笔记volatile键字

    摘要:目前看的部分主要是这个关键字。语言提供了,保证了所有线程能看到共享变量最新的值。前缀的指令在多核处理器下会做两件事情将当前处理器缓存行的数据写回到系统内存。 这一章节的话,主要是讲一下在并发操作中常见的volatile、synchronized以及原子操作的相关知识。 目前看的部分主要是volatile这个关键字。 volatile 根据Java语言规范第3版中对volatile的定义...

    szysky 评论0 收藏0
  • Java并发编程艺术二章读书笔记原子操作

    摘要:前言今天的笔记来了解一下原子操作以及中如何实现原子操作。概念原子本意是不能被进一步分割的最小粒子,而原子操作意为不可被中断的一个或一系列操作。处理器实现原子操作处理器会保证基本内存操作的原子性。 showImg(https://segmentfault.com/img/bVVIRA?w=1242&h=536); 前言 今天的笔记来了解一下原子操作以及Java中如何实现原子操作。 概念 ...

    olle 评论0 收藏0
  • Java并发编程艺术】第一章读书笔记

    摘要:前言并发编程的目的是让程序跑的更快,但并不是启动更多的线程,这个程序就跑的更快。尽可能降低上下文切换的次数,有助于提高并发效率。死锁并发编程中的另一挑战是死锁,会造成系统功能不可用。 前言 并发编程的目的是让程序跑的更快,但并不是启动更多的线程,这个程序就跑的更快。有以下几种挑战。 挑战及方案 上下文切换 单核CPU上执行多线程任务,通过给每个线程分配CPU时间片的方式来实现这个机制。...

    马忠志 评论0 收藏0
  • 那些年我看过书 —— 致敬我大学生活 —— Say Good Bye !

    摘要:开头正式开启我入职的里程,现在已是工作了一个星期了,这个星期算是我入职的过渡期,算是知道了学校生活和工作的差距了,总之,尽快习惯这种生活吧。当时是看的廖雪峰的博客自己也用做爬虫写过几篇博客,不过有些是在前人的基础上写的。 showImg(https://segmentfault.com/img/remote/1460000010867984); 开头 2017.08.21 正式开启我...

    xiaoqibTn 评论0 收藏0
  • 并发编程艺术

    摘要:假设不发生编译器重排和指令重排,线程修改了的值,但是修改以后,的值可能还没有写回到主存中,那么线程得到就是很自然的事了。同理,线程对于的赋值操作也可能没有及时刷新到主存中。线程的最后操作与线程发现线程已经结束同步。 很久没更新文章了,对隔三差五过来刷更新的读者说声抱歉。 关于 Java 并发也算是写了好几篇文章了,本文将介绍一些比较基础的内容,注意,阅读本文需要一定的并发基础。 本文的...

    curlyCheng 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<