zoukankan      html  css  js  c++  java
  • 三、synchronized & lock

    一、锁的类型

    1.1 可重入锁

    • 在执行对象中所有同步方法不用再次获得锁。
    • 如果锁具备可重入性,则称作为可重入锁。synchronized 和 ReentrantLock 都是可重入锁。
    • 可重入性表明了锁的分配机制:基于线程的分配,而不是基于方法调用的分配。
    • 比如说,当一个线程执行到 method1 的 synchronized 方法时,而在 method1 中会调用另外一个 synchronized 方法 method2,此时该线程不必重新去申请锁,而是可以直接执行方法 method2。

    1.2 读写锁

    • 对资源读取和写入的时候拆分为2部分处理,读的时候可以多线程一起读,写的时候必须同步地写。
    • 读写锁将对一个资源的访问分成了2个锁,如文件,一个读锁和一个写锁。正因为有了读写锁,才使得多个线程之间的读操作不会发生冲突。
    • ReadWriteLock 就是读写锁,它是一个接口,ReentrantReadWriteLock 实现了这个接口。可以通过 readLock() 获取读锁,通过 writeLock() 获取写锁。

    1.3 可中断锁

    • 在等待获取锁过程中可中断。
    • 在Java中,synchronized 不是可中断锁,而 Lock 是可中断锁。
    • 如果某一线程 A 正在执行锁中的代码,另一线程 B 正在等待获取该锁,可能由于等待时间过长,线程 B 不想等待了,想先处理其他事情,我们可以让它中断自己或者在别的线程中中断它,这种就是可中断锁。
    • Lock接口中的 lockInterruptibly() 方法就体现了Lock的可中断性。

    1.4 公平锁

    • 按等待获取锁的线程的等待时间进行获取,等待时间长的具有优先获取锁权利。
    • 非公平锁即无法保证锁的获取是按照请求锁的顺序进行的,这样就可能导致某个或者一些线程永远获取不到锁。
    • synchronized 是非公平锁,它无法保证等待的线程获取锁的顺序。对于 ReentrantLock 和 ReentrantReadWriteLock,默认情况下是非公平锁,但是可以设置为公平锁。
    • 而公平锁由于有挂起和恢复所以存在一定的开销,因此性能不如非公平锁,所以 ReentrantLock 和 synchronized 默认都是非公平锁的实现方式。
    // 无参的构造函数创建了一个非公平锁,用户也可以根据第二个构造函数,设置一个 boolean 类型的值,来决定是否使用公平锁来实现线程的调度。
    public ReentrantLock() {
        sync = new NonfairSync(); // 非公平锁
    }
    public ReentrantLock(boolean fair) {
        sync = fair ? new FairSync() : new NonfairSync();
    }
    

    二、Lock

    2.1 Lock 常用方法

    public interface Lock {
    
        /**
         * 用来获取锁,如果锁被其他线程获取,处于等待状态。如果采用Lock,必须主动去释放锁,在发生异常时,不会自动释放锁。
         * 所以,使用Lock必须在try{}catch{}块中进行,并且将释放锁的操作放在finally块中进行,以保证锁一定被被释放,防止死锁的发生。
         */
        void lock();
    
        /**
         * 通过这个方法去获取锁时,如果线程正在等待获取锁,则这个线程能够响应中断,即中断线程的等待状态,可以去做别的事。
         */
        void lockInterruptibly() throws InterruptedException;
    
        /**
         * 表示用来尝试获取锁,如果获取成功,则返回true,如果获取失败(即锁已被其他线程获取),则返回false。
         * 这个方法无论如何都会立即返回。在拿不到锁时不会一直在那等待。
         */
        boolean tryLock();
    
        /**
         * 与tryLock类似,只不过是有等待时间,在等待时间内获取到锁返回true,超时返回false。
         */
        boolean tryLock(long time, TimeUnit unit) throws InterruptedException;
    
        /**
         * 释放锁,在finally 语句块中执行。
         */
        void unlock();
    }
    

    Demo:

    public class LockTest {
        private Lock lock = new ReentrantLock();
    
        /**
         * 测试 lock
         * 结果:
         * 线程名t2获得了锁
         * 线程名t2释放了锁
         * 线程名t1获得了锁
         * 线程名t1释放了锁
         */
        private void method(Thread thread){
            lock.lock();
            try {
                System.out.println("线程名"+thread.getName() + "获得了锁");
            }catch(Exception e){
                e.printStackTrace();
            } finally {
                lock.unlock();
                System.out.println("线程名"+thread.getName() + "释放了锁");
            }
        }
    
        /**
         * 测试 tryLock()
         * 结果:
         * 我是t2有人占着锁,我就不要啦
         * 线程名t1获得了锁
         * 线程名t1释放了锁
         */
        private void method2(Thread thread){
    
            if(lock.tryLock()){
                try {
                    System.out.println("线程名"+thread.getName() + "获得了锁");
                }catch(Exception e){
                    e.printStackTrace();
                } finally {
                    System.out.println("线程名"+thread.getName() + "释放了锁");
                    lock.unlock();
                }
            }else{
                System.out.println("我是"+Thread.currentThread().getName()+"有人占着锁,我就不要啦");
            }
        }
    
    
        public static void main(String[] args) {
            LockTest lockTest = new LockTest();
    
            //线程1
            Thread t1 = new Thread(new Runnable() {
    
                @Override
                public void run() {
                    lockTest.method2(Thread.currentThread());
                }
            }, "t1");
    
            Thread t2 = new Thread(new Runnable() {
    
                @Override
                public void run() {
                    lockTest.method2(Thread.currentThread());
                }
            }, "t2");
    
            t1.start();
            t2.start();
        }
    }
    
    

    2.2 synchronized 和 lock 的对比

    • Lock 是一个接口,而 synchronized 是Java中的关键字,synchronized 是内置的语言实现;
    • synchronized 在发生异常时,会自动释放线程占有的锁,因此不会导致死锁现象发生;而 Lock 在发生异常时,如果没有主动通过 unLock() 去释放锁,则很可能造成死锁现象,因此使用 Lock 时需要在 finally 块中释放锁;
    • ReentrantLock 可设置为公平锁,而 synchronized 却不行;
    • Lock 可以让等待锁的线程响应中断,而 synchronized 却不行,使用 synchronized 时,等待的线程会一直等待下去,不能够响应中断;
    • 通过 Lock 可以知道有没有成功获取锁,而 synchronized 却无法办到,lock 相比 synchronized 更加灵活;
    • Lock 有多个子类,功能较为丰富。
  • 相关阅读:
    poi 导出文件
    获取哪一年 周一的所有日期
    线程池配置
    mybatis基于唯一索引插入或更新
    mongoTemplate关联查询
    cas认证机制
    SpringBoot服务
    HashMap的底层实现
    maven仓库提示“Downloading: http://repo.maven.apache.org/maven2/”
    Tomcat安装SSL证书
  • 原文地址:https://www.cnblogs.com/xiexiandong/p/12910679.html
Copyright © 2011-2022 走看看