sync和lock的区别

97 阅读2分钟

本文已参与「新人创作礼」活动, 一起开启掘金创作之路。

题目: synchronized 和ock有什么区别?用新的Lock有什么好处?你举例说说

1原始构成

synchronized是关键字属于JVM层面,monitorenter(底层是通过monitor对象来完成,其实wait/notify等方法也依赖Fmonitor对象只有在同步块或方法中才能漏wait/notify等方monitorexit

Lock是具体类(java. util. concurrent. locks . Lock)是api层面的锁

2使用方法

synchronized不需要用户去手动释放锁,|当synchronized代码执 行完后系统会自动让线程释放对锁的占用 ReentrantLock,则需要用户去手动释放锁若没有主动释放锁,就有可能导致出现死锁现象。

需要lock() unlock()方法配合try/finally语句块来完成。

3等待是否可中断

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

ReentrantLock可中断,1. 设置超时方法tryLock(long timeout, TimeUnit unit)

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

加锁是否公平. synchronized非公平锁

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

锁绑定多个条件Condition

synchronized没有

ReentrantLock用来实现分组唤醒需要唤醒的线程们,可以精确唤醒, 而不是像synchronized 要么随机唤醒一个线程要 么唤醒全部线程。 sync两次退出,第一次正常退出第二次异常退出

image.png

image.png


package juc;

import sun.awt.windows.ThemeReader;

import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.ReentrantLock;

        class ShareResource{

            private int flag = 1;

            private ReentrantLock lock = new ReentrantLock();

            private Condition condition1 = lock.newCondition();

            private Condition condition2 = lock.newCondition();

            private Condition condition3 = lock.newCondition();

            //ctrl+alt+t是try catch

            public void print5(int totalLoop){
                lock.lock();
                try {
                    //1.判断
                    while (flag != 1){
                        condition1.await();
                    }
                    //干活

                    for (int i = 1;i <= 5; i++) {
                        System.out.println("线程名字:"+Thread.currentThread().getName()+"\t次数:"+i+"\t轮数:"+totalLoop);
                    }

                    //唤醒+通知
                    flag = 2;
                    condition2.signal();

                } catch (Exception e) {
                    e.printStackTrace();
                } finally {
                    lock.unlock();
                }
            }

            public void print10(int totalLoop){
                lock.lock();
                try {
                    //1.判断
                    while (flag != 2){
                        condition2.await();
                    }
                    //干活

                    for (int i = 1;i <= 10; i++) {
                        System.out.println("线程名字:"+Thread.currentThread().getName()+"\t次数:"+i+"\t轮数:"+totalLoop);
                    }

                    //唤醒+通知
                    flag = 3;
                    condition3.signal();

                } catch (Exception e) {
                    e.printStackTrace();
                } finally {
                    lock.unlock();
                }
            }

            public void print15(int totalLoop){
                lock.lock();
                try {
                    //1.判断
                    while (flag != 3){
                        condition3.await();
                    }
                    //干活

                    for (int i = 1;i <= 15; i++) {
                        System.out.println("线程名字:"+Thread.currentThread().getName()+"\t次数:"+i+"\t轮数:"+totalLoop);
                    }

                    //唤醒+通知
                    flag = 1;
                    condition1.signal();

                } catch (Exception e) {
                    e.printStackTrace();
                } finally {
                    lock.unlock();
                }
            }



        }

        public class ConditionLockDemo {
            public static void main(String[] args) {

                ShareResource sr = new ShareResource();

                new Thread(()->{
                    for(int i = 1 ; i<=10 ; i++){
                        sr.print5(i);
                    }
                },"A").start();

                new Thread(()->{
                    for(int i = 1 ; i<=10 ; i++){
                        sr.print10(i);
                    }
                },"B").start();

                new Thread(()->{
                    for(int i = 1 ; i<=10 ; i++){
                        sr.print15(i);
                    }
                },"C").start();
            }
        }