Java并发之synchronized用法及原理

一、synchronized简介

能够保证在同一时刻最多只有一个线程执行该段代码,以达到保证并发安全的效果(当被sync修饰之后,将会以原子方式执行)

地位

  • 是Java的关键字,被Java语言原生支持
  • 是最基本的同步互斥手段
  • 是并发编程中的元老级角色,是并发编程的必学内容

二、synchronized两种用法

1、对象锁

对象锁:包括方法锁(默认锁对象为this当前实例对象)和同步代码块锁(自己制定锁对象)

  • 代码块形式:手动指定锁对象

  • 方法锁形式:synchronized修饰普通方法,锁对象默认为this

生命周期

Debug调试:


Debug调试查看线程所处状态.png
代码实战
  • 代码块形式:
/**
 * 对象锁:代码块
 */
public class SynchronizedObjectCodeBlock02 implements Runnable {

    static SynchronizedObjectCodeBlock02 instance = new SynchronizedObjectCodeBlock02();

    Object lock1 = new Object();
    Object lock2 = new Object();

    @Override
    public void run() {
        // 同一个对象,则运行结果同this
        synchronized (lock1) {

            System.out.println("我是对象锁的代码块形式,我是Lock1,我叫" + Thread.currentThread().getName());
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName() + "Lock1部分,运行结束。");
        }

        // 两把锁会出现并行,若为同一把锁,则串行
        synchronized (lock2) {

            System.out.println("我是对象锁的代码块形式,我是Lock2,我叫" + Thread.currentThread().getName());
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName() + "Lock2部分,运行结束。");
        }
    }

    public static void main(String[] args) {
        Thread t1 = new Thread(instance);
        Thread t2 = new Thread(instance);
        t1.start();
        t2.start();
        while(t1.isAlive() || t2.isAlive()) {

        }
        System.out.println("Finished");
    }
}

  • 方法锁形式
/**
 * 对象锁:代码块
 */
public class SynchronizedObjectMethod03 implements Runnable {

    static SynchronizedObjectMethod03 instance = new SynchronizedObjectMethod03();

    Object lock1 = new Object();
    Object lock2 = new Object();

    @Override
    public void run() {
        try {
            method();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    public synchronized void method() throws InterruptedException {
        System.out.println("我是对象所得方法修饰符形式,我叫" + Thread.currentThread().getName());
        Thread.sleep(3000);
        System.out.println(Thread.currentThread().getName() + ",运行结束。");
    }

    public static void main(String[] args) {
        Thread t1 = new Thread(instance);
        Thread t2 = new Thread(instance);
        t1.start();
        t2.start();
        while(t1.isAlive() || t2.isAlive()) {

        }
        System.out.println("Finished");
    }
}

2、类锁

类锁:指synchronized修饰静态的方法或指定锁对象为Class对象

概念
  • 只有一个Class对象:Java类可能有很多歌对象但是只有一个Class对象
  • 本质:所以所谓的类锁,不过是Class对象的锁而已
  • 用法和效果:类锁只能在同一时刻被一个对象拥有
两种形式
  • synchronized加在static方法上
/**
 * 类锁的第一种形式,static形式
 */
public class SynchronizedClassStatic04 implements Runnable {
    static SynchronizedClassStatic04 instance1 = new SynchronizedClassStatic04();
    static SynchronizedClassStatic04 instance2 = new SynchronizedClassStatic04();

    @Override
    public void run() {
        try {
            method();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    /**
     * 此处不加static,会并行,加static会顺序串行执行
     * @throws InterruptedException
     */
    public static synchronized void method() throws InterruptedException {
        System.out.println("我是类锁的第一种形式static形式,我叫" + Thread.currentThread().getName());
        Thread.sleep(3000);
        System.out.println(Thread.currentThread().getName() + ",运行结束。");
    }

    public static void main(String[] args) {
        Thread t1 = new Thread(instance1);
        Thread t2 = new Thread(instance2);
        t1.start();
        t2.start();
        while(t1.isAlive() || t2.isAlive()) {

        }
        System.out.println("Finished");
    }
}

  • synchronized(*.class)代码块
/**
 * 类锁的第一种形式,static形式
 */
public class SynchronizedClassClass05 implements Runnable {
    static SynchronizedClassClass05 instance1 = new SynchronizedClassClass05();
    static SynchronizedClassClass05 instance2 = new SynchronizedClassClass05();

    @Override
    public void run() {
        try {
            method();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    public void method() throws InterruptedException {
        /**
         * 此处若为this,则并行,*.class则串行
         */
        synchronized (SynchronizedClassClass05.class) {

            System.out.println("我是类锁的第二种形式static形式:synchronized(*.class),我叫" + Thread.currentThread().getName());
            Thread.sleep(3000);
            System.out.println(Thread.currentThread().getName() + ",运行结束。");
        }
    }

    public static void main(String[] args) {
        Thread t1 = new Thread(instance1);
        Thread t2 = new Thread(instance2);
        t1.start();
        t2.start();
        while(t1.isAlive() || t2.isAlive()) {

        }
        System.out.println("Finished");
    }
}

三、性质

可重入性

指的是同一线程的外层函数获得锁之后,内层函数可以直接再次获取该锁

好处:避免死锁,提升封装性

粒度:线程而非调用(用3种情况来说明和pthread的区别)
3种情况:
1、证明同一个方法是可重入的
2、证明可重入不要求是同一个方法
3、证明可重入不要求是同一个类中的

不可中断性

一旦这个锁被别人获得了,如果我还想获得,我只能选择等待或者阻塞,直到别的线程释放这个锁,如果别人永远不释放锁,那么我只能永远的等下去

相比之下,Lock类,可以拥有中断的能力,第一点:如果我觉得我等的时间太长了,有权中断现在已经获取到锁的线程的执行;第二点:如果我觉得我等的时间太长了,不想再等了,可以退出

四、Synchronized原理

加锁和释放锁的原理:现象、时机、深入JVM看字节码
可重入原理:加锁次数计数器
保证可见性的原则:内存模型

1、加锁和释放锁的原理

  • 现象
  • 获取和释放锁的时机:内置锁
  • 等价代码
  • 深入JVM看字节码:反编译看monitor指令
深入JVM看字节码
  • 概况
/**
 * 反编译字节码
 */
public class Decompilation4 {
    private Object object = new Object();

    public void insert(Thread thread) {
        synchronized (object) {

        }
    }
}

  • 如何反编译
javac Decompilation4.java
javap -verose Decompilation4.class

看到如下条目6、8、14


image.png
  • monitorenter和monitorexit指令

2、可重入原理:加锁次数计数器

  • JVM负责跟踪对象被加锁的次数
  • 线程第一次给对象加锁的时候,计数变为1.每当这个相同的线程在此对象上在此获得锁时,计数会递增
  • 每当任务离开时,计数递减,当计数为0时,这把锁被完全释放

3、可见性的原理:Java内存模型

线程间通信图.png

Synchronized修饰的代码块,在锁释放之前写入到主内存中,保证了本地内存和主内存一致性。

五、Synchronized缺陷

  • 效率低:锁的释放情况少、试图获得锁时不能设定超时,不能中断一个正在糊涂获得锁的线程

  • 不够灵活(读写锁更灵活):加锁和释放的时机单一,每个锁仅有单一的条件(某个对象),可能是不够的

  • 无法直到是否成功获取到锁

以上内容整理自网络

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

推荐阅读更多精彩内容