Synchronized 和 Lock的区别,以及使用Lock的好处

Synchronized 和 Lock 在java并发编程中广泛使用,简单的来说下这两者的区别,记录下

一,两者的构造

  synchronized是关键字,它是属于JVM层面的

  Lock是一个具体的类,它是属于API层面的锁  (java.util.concurrent.locks.Lock)

synchronized底层是通过monitor对象来完成

二,使用方法

  synchronized 不需要用户手动去释放锁,当synchronized代码执行完后系统会自动让线程释放对锁的占用

  ReentrantLock则需要用户去手动释放锁,若没有主动释放锁,可能会导致死锁现象。需要lock()和unlock()方法配合try/finally语句块来完成

三,等待是否可以中断

  synchronized不可中断,除非抛出异常或者正常运行完成

  ReentrantLock 可中断:

    1),设置超时方法tryLock(long timeout,TimeUnit unit)

    2),lockInterruptibly()放代码块中,调用interrupt()方法可中断

推荐查看此博客:https://www.cnblogs.com/dolphin0520/p/3923167.html

四,加锁是否公平

  synchronized 非公平锁

  ReentrantLock两者都可以,默认非公平锁,构造方法可以传入boolean值,true为公平锁,false为非公平锁

  

五,锁绑定多个条件Condition

  synchronized 没有

  ReentrantLock用来实现分组唤醒需要唤醒的线程们,可以精确唤醒,而不是像synchronized要么随机唤醒一个要么唤醒全部线程。

Lock的好处,在第五点看来绑定多个Condition  可以精确唤醒,用一下列子来看看:

/**
 *
 * 编写一个线程,开启三个线程,这三个线程的ID分别为A,B,C,每个线程将自己的ID在屏幕上打印10遍,要求输出结果必须按顺序显示。
 * 如:ABCABCABC... 依次递归
 *
 */
public class TestABCAlternate {

    public static void main(String[] args) {
        AlternateDemo ad = new AlternateDemo();

        new Thread(new Runnable() {
            public void run() {

                for (int i = 1; i <= 20; i++) {
                    ad.loopA(i);
                }

            }
        },"A").start();

        new Thread(new Runnable() {
            public void run() {

                for (int i = 1; i <= 20; i++) {
                    ad.loopB(i);
                }

            }
        },"B").start();

        new Thread(new Runnable() {
            public void run() {

                for (int i = 1; i <= 20; i++) {
                    ad.loopC(i);

                    System.out.println("===========================");
                }
            }
        },"C").start();
    }
}

class AlternateDemo{
    private int number = 1;//当前正在执行线程的标记

    private Lock lock = new ReentrantLock();

    private Condition condition1 = lock.newCondition();  //多个condition 精确唤醒
    private Condition condition2 = lock.newCondition();
    private Condition condition3 = lock.newCondition();
    /**
     *
     * @param totalLoop : 循环第几轮
     */
    public void loopA(int totalLoop) {
        lock.lock();
        try {
            //1.判断
            while(number!=1) {    //使用while  不使用if  是防止虚假唤醒
                condition1.await();
            }
            //2.打印
            for (int i = 1; i <= 1; i++) {
                System.out.println(Thread.currentThread().getName()+"\t"+i+"\t"+totalLoop);
            }
            //3.唤醒
            number = 2;
            condition2.signal();
        } catch (Exception e) {
            // TODO: handle exception
        } finally {
            lock.unlock();
        }
    }
    public void loopB(int totalLoop) {
        lock.lock();
        try {
            //1.判断
            while(number!=2) {
                condition2.await();
            }
            //2.打印
            for (int i = 1; i <= 1; i++) {
                System.out.println(Thread.currentThread().getName()+"\t"+i+"\t"+totalLoop);
            }
            //3.唤醒
            number = 3;
            condition3.signal();
        } catch (Exception e) {
            // TODO: handle exception
        } finally {
            lock.unlock();
        }
    }
    public void loopC(int totalLoop) {
        lock.lock();
        try {
            //1.判断
            while(number!=3) {
                condition3.await();
            }
            //2.打印
            for (int i = 1; i <= 1; i++) {
                System.out.println(Thread.currentThread().getName()+"\t"+i+"\t"+totalLoop);
            }
            //3.唤醒
            number = 1;
            condition1.signal();
        } catch (Exception e) {
            // TODO: handle exception
        } finally {
            lock.unlock();
        }
    }
}

原文地址:https://www.cnblogs.com/lxwt/p/11345815.html

时间: 2024-10-27 17:06:01

Synchronized 和 Lock的区别,以及使用Lock的好处的相关文章

Java中synchronized和Lock的区别

synchronized和Lock的区别synchronize锁对象可以是任意对象,由于监视器方法必须要拥有锁对象那么任意对象都可以调用的方法所以将其抽取到Object类中去定义监视器方法这样锁对象和监视器对象是同一个,只要创建了锁对象它既是锁对象同时也是监视器对象这样不能实现在一个锁对象上绑定多个监视器对象jdk1.5中Lock对象仅仅是一个锁对象监视器方法被封装到了Condition对象中这样实现了锁对象和监视器对象进行了分离更加的面向对象这样可以实现在一个锁对象上绑定多个监视器对象 在一个

synchronized和java.util.concurrent.locks.Lock的区别

在看ConcurrentHashMap 源码的时候看到lock这个锁机制,不明白它和Synchronized的区别,查了一些资料记录下来,在Lock的文档中,对Lock的解释是:Lock实现比synchronized 提供了更多额外的锁操作,它有更灵活的结构,可以支持不同的属性,可以支持多个相关条件的对象.那下边看一下Lock可以提供哪些比synchronized 额外的操作,也就是解决synchronized 存在的问题: 1.它无法中断一个正在等候获得锁的线程 2.也无法通过投票得到锁,如果

Synchronized和lock的区别和用法

一.synchronized和lock的用法区别 (1)synchronized(隐式锁):在需要同步的对象中加入此控制,synchronized可以加在方法上,也可以加在特定代码块中,括号中表示需要锁的对象. (2)lock(显示锁):需要显示指定起始位置和终止位置.一般使用ReentrantLock类做为锁,多个线程中必须要使用一个ReentrantLock类做为对 象才能保证锁的生效.且在加锁和解锁处需要通过lock()和unlock()显示指出.所以一般会在finally块中写unloc

synchronized和lock的区别和使用

---恢复内容开始--- 1.线程与进程 一个程序至少需要一个线程,一个进程至少需要一个线程  线程->进程->程序 线程是程序执行流的最小单位,进程是系统进行资源分配和调度的一个独立单位. 2.Thread的几个重要方法 ①start()方法:开始执行该线程 ②stop()方法:强制结束该线程 ③join()方法 :等待该线程结束 ④sleep()方法:该线程进入等待 ⑤run()方法    :直接执行该线程的run方法(线程调用start()也会执行run方法,区别是一个是由线程调度运行r

java基础: synchronized与Lock的区别

主要区别 1. 锁机制不一样:synchronized是java内置关键字,是在JVM层面实现的,系统会监控锁的释放与否,lock是JDK代码实现的,需要手动释放,在finally块中释放.可以采用非阻塞的方式获取锁: 2. 性能不一样:资源竞争激励的情况下,lock性能会比synchronize好,竞争不激励的情况下,synchronize比lock性能好,synchronize会根据锁的竞争情况,从偏向锁-->轻量级锁-->重量级锁升级,而且编程更简单 3. synchronized无法判

synchronized 与 lock 的区别

synchronized 和 lock 的用法区别 synchronized(隐式锁):在需要同步的对象中加入此控制,synchronized 可以加在方法上,也可以加在特定代码块中,括号中表示需要锁的对象. lock(显示锁):需要显示指定起始位置和终止位置.一般使用 ReentrantLock 类做为锁,多个线程中必须要使用一个 ReentrantLock 类做为对象才能保证锁的生效.且在加锁和解锁处需要通过 lock() 和 unlock() 显示指出.所以一般会在 finally 块中写

java - synchronized与lock的区别

synchronized与lock的区别 原始构成 synchronized是关键字属于JVM层面 monitorenter(底层是通过monitor对象来完成,其实wait/notify等对象也依赖于monitor独享只有在同步块或方法中才能调wait/notify等方法) monitorexit Lock是具体类(java.utl.concurrent.locks.Lock)是api层面的锁 使用方法 synchronized不需要用户去手动释放锁,当synchronized代码执行完后系统

深入Synchronized和java.util.concurrent.locks.Lock的区别详解

主要相同点:Lock能完成Synchronized所实现的所有功能.主要不同点:Lock有比Synchronized更精确的线程予以和更好的性能.Synchronized会自动释放锁,但是Lock一定要求程序员手工释放,并且必须在finally从句中释放.synchronized 修饰方法时 表示同一个对象在不同的线程中 表现为同步队列如果实例化不同的对象 那么synchronized就不会出现同步效果了.1.对象的锁 所有对象都自动含有单一的锁. JVM负责跟踪对象被加锁的次数.如果一个对象被

synchronized和lock的区别

原文摘自:https://www.eyesmoons.com/article/75 1,原始构成 synchronized是关键字,属于JVM层面,通过wait,notify和notifyAll来调度线程. Lock是具体类,是api层面的锁. 2,使用方法 synchronized不需要用户手动去释放锁, 当synchronized代码执行完后,系统会自动释放锁. Lock需要用户手动释放锁,否则会出现死锁现象.需要lock和unlock配合try/finally语句块来完成. 3,等待是否中