Java锁(一):volatile、synchronized详解

一、锁的基础知识

锁的类型

锁从客观上分为悲观锁和乐观锁。

  • 乐观锁:乐观锁是一种乐观思想,认为写少读多,遇到并发写的可能性比较低,读数据的时候认为别人不会修改,所以读的时候不会上锁,但是在写的时候会判断一下在此期间有没有别人去更新这个数据,采取的是先读取当前版本号,然后加锁操作,写完的时候读取最新版本号做记录的版本号做比较一样则成功,如果失败则重复读-比较-写的操作。Java中的乐观锁基本都是通过CAS操作实现的,java.util.concurrent.atomic包下的原子变量。CAS(compare and swap)比较交换是一种更新的原子操作,比较当前值和传入值是否一样,一样则更新,否则则失败。
  • 悲观锁:悲观锁就是悲观思想,认为写多且遇到并发性的可能性高,每次拿数据的时候都认为别人为修改,所以每次读写的时候都会上锁,这样别人想读写数据的时候都会block(阻塞)知道拿到锁。Java中悲观锁就是syschronizedAQS框架下的锁则是先尝试CAS乐观锁获取锁,如果获取不到,才会转为悲观锁,如ReentrantLock

Java中的锁

在Java中主要有两种锁加锁机制:

  • syschronized关键字修饰
  • java.util.concurrent.Lock,Lock是一个接口,有很多实现类比如ReentrantLock

二、volatile

可见性

public class VolatileTest {
    public static void main(String[] args) {
        final  VT vt = new VT();
        Thread thread01 = new Thread(vt);
        Thread thread02 = new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    Thread.sleep(3000);
                } catch (InterruptedException ignore) { }
                vt.sign = true;
                System.out.println("vt.sign = true 通知 while (!sign) 结束!");
            }
        });
        thread01.start();
        thread02.start();
    }
}

class VT implements Runnable {
    public boolean sign = false;
    @Override
    public void run() {
        while (!sign) {

        }
        System.out.println("你坏");
    }
}

上面的代码是两个线程同时操作一个变量,程序希望当sign在线程Thread01被操作vt.sign = true时,线程Thread02输出你坏

实际上这段代码永远不会输出你坏,而是一直处于死循环。这是为什么呢?接下来我们一步步讲解验证。

我们把sign关键字加上volatile关键字。

public volatile boolean sign = false;

这个时候会输出你坏

volatile关键字是Java虚拟机提供的最轻量级锁的同步机制,作为一个修饰符出现,同来修饰变量,不含括局部变量,用来保证对所有线程可见性。

volatile关键字修饰时内存变化

16.jpg

当没有volatile关键字修饰的时候,Thread01对变量进行操作,Thead02并不会拿到最新值。

volatile关键字时内存变化

17.jpg

当有volatile关键字修饰的时候,Thread01对变量进行操作时,会把变量的变化强制刷新到主内存,Thread02获取值时,会把自己内存的sign值过期掉,从主内存读取最新的。

有序性

volatile关键字底层是通过lock指令实现可见性的,lock指令相当于一个内存屏障,保证以下三点:

  • 将本处理器的缓存写入主内存。
  • 重排序时不会把后面的指令重新排序到内存屏障之前。
  • 如果是写入操作会导致其他内存器中对应的内存无效。

总结

  • volatile关键字会控制被修饰的变量在内存操作的时候会主动把值刷新到主内存,JMM会先将线程对应的CPU内存设置过期,从内存读取最新值。
  • volatile关键字是通过内存屏障防止指令重排,volatile的内存屏障在读写的时候在前后各添加一个Store屏障来保证重新排序时不会把内存屏障后面的时候指令排序到内存屏障之前。
  • volatile不能解决原子性,如果需要解决原子性需要synchronized或者lock

三、synchronized

知识大纲

18.jpg

使用方法

synchronized关键字主要有以下三种使用方式:

  • 修饰实例方法,作用于当前实例加锁,进入同步代码前要获取当前实例的锁。

    public class SynchronizedTest implements Runnable{
        private static int i = 0;
    
        public synchronized void getI(){
            if (i % 1000000 == 0) {
                System.out.println(i);
            }
        }
    
        public synchronized void increase() {
            i++;
            getI();
        }
    
        @Override
        public void run() {
            for (int j = 0; j < 1000000; j++) {
                increase();
            }
            System.out.println(i);
        }
    
        public static void main(String[] args) {
            ExecutorService executorService = Executors.newCachedThreadPool();
            SynchronizedTest synchronizedTest = new SynchronizedTest();
            executorService.execute(synchronizedTest);
            executorService.execute(synchronizedTest);
            executorService.shutdown();
        }
    }
    

    最后结果输出:

    1000000
    1556623
    2000000
    2000000
    

    上述代码中,创建两个线程同时操作同一个共享资源i,且increase()get()方法加了synchronized关键字,表示当前线程的锁是实例对象,因为传入线程都是synchronizedTest对象实例是同一个,所以最终结果肯定能输出2000000,如果我们换种方式,传入不同对象,代码如下:

    public static void main(String[] args) {
        ExecutorService executorService = Executors.newCachedThreadPool();
        SynchronizedTest synchronizedTest01 = new SynchronizedTest();
        SynchronizedTest synchronizedTest02 = new SynchronizedTest();
        executorService.execute(synchronizedTest01);
        executorService.execute(synchronizedTest02);
        executorService.shutdown();
    }
    

    输出如下:

    1002588
    1641267
    1848269
    

    最终肯定不是期望的200000,因为synchronized修饰方法锁的是当前实例,传入不同对象实例线程是无法保证安全的。

  • 修饰静态方法,作用于当前类对象加锁,进入同步方法前要获取当前类对象的锁。

    public class SynchronizedTest implements Runnable{
        private static int i = 0;
    
        public synchronized static void getI(){
            if (i % 1000000 == 0) {
                System.out.println(i);
            }
        }
    
        public synchronized static void increase() {
            i++;
            getI();
        }
    
        @Override
        public void run() {
            for (int j = 0; j < 1000000; j++) {
                increase();
            }
            System.out.println(i);
        }
    
        public static void main(String[] args) {
            ExecutorService executorService = Executors.newCachedThreadPool();
            SynchronizedTest synchronizedTest01 = new SynchronizedTest();
            SynchronizedTest synchronizedTest02 = new SynchronizedTest();
            executorService.execute(synchronizedTest01);
            executorService.execute(synchronizedTest02);
            executorService.shutdown();
        }
    }
    

    输出如下:

    1000000
    1649530
    2000000
    2000000
    

    上述代码和第一段代码差不多,只不过increase()get()方法是静态方法,且也加上了synchronized表示锁的是当前类对象,虽然我们传入不同的对象,但是最终结果是会输出200000的。

  • 修饰语代码块,指定加锁对象,给对象加锁,进入同步方法前要获取给定对象的锁。

    public class SynchronizedTest02 implements Runnable{
        private static SynchronizedTest02 synchronizedTest02 = new SynchronizedTest02();
        private static int i = 0;
    
        @Override
        public void run() {
            // 传入对象锁当前实例对象
            // 如果是 synchronized (SynchronizedTest02.class) 锁当前类对象
            synchronized (synchronizedTest02){
                for(int j=0;j<1000000;j++){
                    i++;
                }
            }
        }
    
        public static void main(String[] args) throws Exception {
            Thread thread01 = new Thread(synchronizedTest02);
            Thread thread02 = new Thread(synchronizedTest02);
            thread01.start();
            thread02.start();
            Thread.sleep(3000);
            System.out.println(i);
        }
    }
    

    上述代码用锁修饰代码块,传入的是对象表示锁的是当前实例对象,如果传入是类表示锁的是类对象。

特性

原子性

原子性表示一个操作不可中断,要么成功要么失败。

synchroniezd能实现方法同步,同一时间段内只有一个线程能拿到锁,进入到代码执行,从而达到原子性。

底层通过执行mointorenter指令,判断是否有ACC_SYNCHRONIZED同步标识,有表示获取monitor锁,此时计数器+1,方法执行完毕,执行mointorexit指定,此时计数器-1,归0释放锁。

可见性

可见性表示一个线程修改了一个共享变量的值,其它线程都能够知道这个修改。CPU缓存优化指令重排等都可能导致共享变量修不能立刻被其他线程察觉。

synchroniezd通过操作系统内核互斥锁实现可见性,线程释放锁前必须把共享变量的最新值刷新到主内存中,线程获取锁之前会将工作内存中共享值清空,从主内存中获取最新的值。

有序性

程序在执行时,有可能会进行指令重排,CPU执行指令顺序不一定和程序的顺序一致。指定重排保证串行语义一致(即重排后CPU执行的执行和程序真正执行顺序一致)。synchronized能保证CPU执行指令顺序和程序的顺序一致。

public class LazySingleton {

    /**
     * 单例对象
     * volatile + 双重检测机制 -> 禁止重排序
     */
    private volatile static LazySingleton instance = null;

    /**
     *   instance = new LazySingleton();
     *   1. 分配对象内存空间
     *   2. 初始化对象
     *   3. 设置instance指向刚分配的内存
     *
     *   JVM和CPU优化, 发生了指令重排, 1-3-2, 线程A执行完3, 线程B执行第一个判断, 直接返回, 这个时候是     *   有问题的。
     *   通过volatile关键字禁止重排序
     * @return
     */
    public static LazySingleton getInstance(){
        if (null == instance) {
            synchronized (LazySingleton.class){
                if (null == instance) {
                    instance = new LazySingleton();
                }
            }
        }
        return instance;
    }
}

synchronized的有序性是保证线程有序的执行,不是防止指令重排序。上面代码如果不加volatile关键字可能导致的结果,就是第一个线程在初始化的时候,设置instance执行分配的内存时,这个时候第二个线程进来了,有指令重排,在第一个判断的时候直接返回,就出错了这个时候instance可能还没初始化成功。

重入性

synchronized是可重入锁,允许一个线程二次请求自己持有对象锁的临界资源。

public class SynchronizedTest03 extends A {

    public static void main(String[] args) {
        SynchronizedTest03 synchronizedTest03 = new SynchronizedTest03();
        synchronizedTest03.doA();
    }

    public synchronized void doA() {
        System.out.println("子类方法:SynchronizedTest03.doA() ThreadId:" + Thread.currentThread().getId());
        doB();
    }

    public synchronized void doB() {
        System.out.println("子类方法:SynchronizedTest03.doB() ThreadId:" + Thread.currentThread().getId());
        super.doA();
    }
}

class A {
    public synchronized  void doA() {
        System.out.println("父类方法:A.doA() ThreadId:" + Thread.currentThread().getId());
    }
}

上面代码正常输入如下:

子类方法:SynchronizedTest03.doA() ThreadId:1
子类方法:SynchronizedTest03.doB() ThreadId:1
父类方法:A.doA() ThreadId:1

最后正常的输出了结果,并没有发生死锁,说明synchronized是可重入锁。

synchronized锁对象的时候有个计数器,记录线程获取锁的次数,在执行完对应的代码后计数器就会-1,知道计数器清0释放锁。

类型和升级

在介绍锁的类型之前先说一下什么是markwordmarkword是java对象数据结构中的一部分,markword数据在长度为32位和64位虚拟机(未开启压缩指针)中分别是32bit和64bit,它的最后两位bit是锁状态标志位,用来标记当前对象的状态,如下表示:

状态 标志位 储存内容
无锁(未开启偏向锁) 01 对象哈希码、对象分代年龄
偏向锁(开启偏向锁) 01 偏向线程id、偏向时间戳、对象分代年龄
轻量级锁 00 指向轻量级锁指针
重量级锁 10 指向重量级锁指针
GC标记 11
偏向锁

偏向锁会偏向于第一个访问锁的线程,如果在运行过程中只有一个线程访问不存在多个线程争用的情况下,则线程是不需要触发同步的,这个时候就会给线程加一个偏向锁。如果在运行过程中,遇到了其他线程抢占锁,则持有偏向锁的线程会被挂起,JVM会消除它身上的偏向锁,将锁升级至轻量级锁。

UseBiasedLocking 是一个偏向锁检查, 1.6 之后是默认开启的, 1.5 中是关闭的,需要手动开启参数是 XX: UseBiasedLocking=false

偏向锁获取过程:

  1. 访问markword中偏向锁表示是否为1,锁标志位01,确认为偏向锁状态。
  2. 判断markword中线程id是否指向当前线程id,如果是则执行步骤5,如果不是则执行步骤3
  3. 如果markword中线程id未指向当前线程id,则通过CAS操作竞争锁。如果竞争成功,则指向当前线程id,执行步骤5,如果竞争失败,则执行步骤4。
  4. 如果CAS竞争锁失败表示有竞争,当到达全局安全点(safepoint)时获得偏向锁的线程会被挂起,偏向锁升级为轻量级锁并撤销偏向锁(撤销偏向锁是会导致stop the word,除GC所需的线程外,所有的线程都进入等待状态,直到GC任务完成),然后被阻塞在安全点的线程会继续执行同步代码。
  5. 执行同步代码。
轻量级锁

当锁是偏向锁的时候,在运行过程中发现有其他线程抢占锁,偏向锁就会升级成轻量级锁,其他线程会通过自旋的形式获取锁,不会阻塞,提高性能,缺点是循环会消耗CPU。

轻量级锁加锁过程:

  1. 在代码进入同步块的时候,如果同步对象锁状态为无锁状态(锁状态标志位为01状态,是否为偏向锁为0),虚拟机首先将在当前线程的帧栈中建立一个名为索记录(Lock Record)的空间,用于储存锁对象目前的markword的拷贝,官方称之为 Displaced Mark Word
  2. 拷贝对象的markword到锁记录中。
  3. 拷贝成功后,虚拟机将使用CAS操作尝试将对象的markword更新指向锁记录的指针,并将锁记录里的owner指向对象的markword,如果更新成功则执行步骤4,否则执行步骤5。
  4. 更新成功表示这个线程就获取到了锁的对象,并且对象的markword锁标志位设置成00,表示此对象处于轻量级锁 状态。
  5. 如果更新失败了,说明虚拟机首先会检查对象的markword是否指向当前线程的栈帧,如果是说明当前线程已经获取到了这个对象的锁。如果不是则说明多个线程竞争锁,轻量级锁就会升级成重量级锁,锁标志的状态值变为10,markword中储存的就是指向重量级锁的指针,后面等待锁的线程会进入阻塞状态。
重量级锁

当偏向锁升级成轻量级锁时,其他线程会通过自旋的方式获取锁,不会阻塞,如果自旋n次都失败了,这个时候轻量级锁就会升级成重量级锁。

总结
19.jpg

synchronized的执行过程:

  1. 检查markword里面存储的是不是当前线程的id,如果是则表示当前线程处于偏向锁。
  2. 如果不是,则尝试使用CAS将当前线程的id替换markword,如果成功则表示当前线程获取锁,偏向标志位置为1。
  3. 如果CAS失败则说明发生竞争,撤销偏向锁,进而升级成轻量级锁,锁标志置为00。
  4. 当前线程使用CAS将对象的markword替换成锁记录指针,如果成功,则当前线程获取锁。
  5. 如果替换失败,表示其他线程竞争锁,当前线程遍尝试使用自选锁的方式来获取锁。
  6. 如果自旋成功获取锁则依处于轻量级锁。
  7. 如果自旋失败,则升级成重量级锁,锁标志置为10。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,445评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,889评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,047评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,760评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,745评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,638评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,011评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,669评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,923评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,655评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,740评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,406评论 4 320
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,995评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,961评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,023评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,483评论 2 342

推荐阅读更多精彩内容