Java 锁 synchronized VS Lock

更多 Java 并发编程方面的文章,请参见文集《Java 并发编程》


synchronized 关键字

  • 用在普通方法前 synchronized void func():对象锁

    • 等价与 synchronized(this) {...}
  • 用在 static 方法前 synchronized static void func():类锁

    • 等价与 synchronized(A.class) {...}
  • synchronized(obj) {...}:对象锁

synchronized 的问题:

  • 不区分读写。如果多个线程对同一个对象进行读操作,实际上不应该有冲突,但是 synchronized 会导致线程等待。
  • 会导致线程无限等待。不可以中断等待。

Lock 接口 VS synchronized 关键字

  • 类型:
    • Lock 为接口,有不同的实现类
    • synchronized 为关键字
  • 锁的释放:
    • Lock 需要手动释放锁,即 unLock()
    • synchronized 不需要手动释放锁,synchronized 代码块执行完,自动释放锁
  • 等待的中断:
    • Lock 可以中断等待,即 t.interrupt()
    • synchronized 会导致线程无限等待,不可中断
  • Lock 可以知道线程有没有成功获得锁,即 tryLock(),synchronized不支持

Lock 接口

Java 5 的 concurrent 包开始提供。
提供的方法包括:

  • void lock();:获取锁,如果不能成功获取,则等待,与 synchronized 类似
  • void lockInterruptibly();:获取锁,如果不能成功获取,则等待,且能够响应中断 t.interrupt()
  • boolean tryLock();:获取锁,如果成功,返回 true,如果不成功,返回 false,不等待
  • boolean tryLock(long time, TimeUnit unit);在一定的时间内,获取锁,如果成功,返回 true,如果不成功,返回 false,不等待
  • void unlock();:释放锁

可重入锁

可重入锁:锁基于线程分配,而不是基于方法分配。

synchronized 为可重入锁,例如:
在下面的代码中,在线程进入 f1() 方法时已经获得了对象的锁,因此在 f1() 中调用 f2() 时,不需要重新申请锁。

public synchronized void f1() {
  // 不需要重新申请锁
  f2();
}
public synchronized void f2() {
  // to do
}

ReentrantLock 可重入锁

Java 5 的 concurrent 包开始提供,继承了 Lock 接口。
public class ReentrantLock implements Lock, java.io.Serializable {

关于 ReentrantLock 的实现原理,参见 Java AQS AbstractQueuedSynchronizer

下面的例子中提供了两种方式来控制对共享变量 count 的并行读写操作:

  • incrementAndGetCount():使用 ReentrantLock:
    • 记住:unlock() 操作不能忘记,应该放在 finally 中,确保在出现异常状况下也会被调用。
  • incrementAndGetCountWithSynchronized():使用 synchronized 关键字。
public class Lock_Test {
    private Lock lock = new ReentrantLock();

    private int count = 0;

    private int incrementAndGetCount() {
        lock.lock();

        try {
            System.out.println(Thread.currentThread().getName() + " gets Count: " + count);
            return count++;
        } finally {
            lock.unlock();
        }
    }

    private synchronized int incrementAndGetCountWithSynchronized() {
        System.out.println(Thread.currentThread().getName() + " gets Count: " + count);
        return count++;
    }

    public static void main(String[] args) {
        Lock_Test test = new Lock_Test();

        for (int i = 0; i < 5; i++) {
            new Thread() {
                public void run() {
                    test.incrementAndGetCount();

                    try {
                        Thread.sleep(100);
                    } catch (InterruptedException ex) {
                        ex.printStackTrace();
                    }
                }
            }.start();
        }
    }
}

可中断锁

在上文中已经提到过:

  • synchronized 会导致线程无限等待,不可中断
  • Lock 可以中断等待,即 t.interrupt(),示例如下:
    线程 t1 获得了锁,但是并没有在 finally 中释放锁,因此可以通过 t2.interrupt(); 使得线程 t2 停止等待。
    注意:想要能够响应中断,需使用 lock.lockInterruptibly(); 而不能是 lock.lock();
public class Lock_Test2 {
    private Lock lock = new ReentrantLock();

    private int count = 0;

    private int incrementAndGetCount() {
        try {
            // 想要能够响应中断,需使用 lock.lockInterruptibly(); 而不能是 lock.lock();
            lock.lockInterruptibly();

            System.out.println(Thread.currentThread().getName() + " gets Count: " + count);
            return count++;
        } catch (Exception e) {
            return 0;
        } finally {
            // 并没有在 finally 中释放锁
            // lock.unlock();
        }
    }

    public static void main(String[] args) {
        Lock_Test2 test = new Lock_Test2();

        Thread t1 = new Thread() {
            public void run() {
                test.incrementAndGetCount();

                try {
                    Thread.sleep(100);
                } catch (InterruptedException ex) {
                    ex.printStackTrace();
                }
            }
        };

        Thread t2 = new Thread() {
            public void run() {
                test.incrementAndGetCount();

                try {
                    Thread.sleep(100);
                } catch (InterruptedException ex) {
                    ex.printStackTrace();
                }
            }
        };

        t1.start();
        t2.start();
        t2.interrupt();
    }
}

公平锁

  • 公平锁:加锁前检查是否有排队等待的线程,先来先得 FIFO,即先排队,再尝试获取锁
  • 非公平锁:加锁时不考虑排队等待问题,直接尝试获取锁,获取不到自动到队尾等待,即先尝试获取锁,再排队

ReentrantLock 默认是非公平锁,可以通过构造方法设置为公平锁:

    public ReentrantLock(boolean fair) {
        sync = fair ? new FairSync() : new NonfairSync();
    }

公平锁可以被用来解决 饥饿 问题。什么是 饥饿 问题?举例说明:

  • T1 锁定资源
  • T2 请求资源,T2 等待
  • T3 请求资源,T3 等待
  • T1 释放资源,T3 获得
  • T4 请求资源,T4 等待
  • T3 释放资源,T4 获得
  • T2 可能永远等待,出现饥饿

读写锁

  • 读锁 被占用,则申请写锁的线程会等待,申请读锁的线程不用等待
  • 写锁 被占用,则申请写锁或读锁的线程都会等待

ReadWriteLock 为读写锁的接口,ReentrantReadWriteLock 为一个实现类。

  • 通过 Lock readLock(); 方法得到读锁
  • 通过 Lock writeLock(); 方法得到写锁

在下面的例子中,输出如下:

Thread-4 is reading count: 0
Thread-0 is reading count: 0
Thread-3 is reading count: 0
Thread-2 is reading count: 0
Thread-1 is reading count: 0
Thread-5 is writing count: 0
Thread-6 is writing count: 1
Thread-7 is writing count: 2
Thread-8 is writing count: 3
Thread-9 is writing count: 4

可以看出:

  • 在某个线程获得读锁后,其他申请读锁的线程不需等待,而其他申请写锁的线程需要等待。因此 writing 都在 reading 之后。
  • 在某个线程获得写锁后,其他申请读锁或者写锁的线程都需要等待。因此 writing 按照 5,6,7,8,9 的顺序依次执行。
public class ReadWriteLock_Test {
    private static final ReentrantReadWriteLock lock = new ReentrantReadWriteLock();

    private static int count = 0;

    public static void readCount() {
        // 申请读锁
        lock.readLock().lock();
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
        }
        System.out.println(Thread.currentThread().getName() + " is reading count: " + count);
        // 申请读锁
        lock.readLock().unlock();
    }

    public static void writeCount() {
        // 申请写锁
        lock.writeLock().lock();
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
        }
        System.out.println(Thread.currentThread().getName() + " is writing count: " + count++);
        // 释放写锁
        lock.writeLock().unlock();
    }

    public static void main(String[] args) {
        for (int i = 0; i < 5; i++) {
            (new ReadThread()).start();
        }

        for (int i = 0; i < 5; i++) {
            (new WriteThread()).start();
        }
    }

    static class ReadThread extends Thread {
        public void run() {
            readCount();
        }
    }

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

推荐阅读更多精彩内容