三、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 有多个子类,功能较为丰富。
原文地址:https://www.cnblogs.com/xiexiandong/p/12910679.html