本文参考自:Java并发编程:Lock。

Lock锁并不是java内置的功能,其应用场景是在多线程并发访问时,为了避免冲突,需要每个线程先获取锁,避免其他线程的进入,等线程执行完后释放锁,允许其他线程进入。

1. Lock锁与synchronized同步的区别

  • Lock是一个接口,而synchronized是Java中的关键字,synchronized是内置的语言实现;
  • synchronized在发生异常时,会自动释放线程占有的锁,因此不会导致死锁现象发生;而Lock在发生异常时,如果没有主动通过unLock()去释放锁,则很可能造成死锁现象,因此使用Lock时需要在finally块中释放锁;
  • Lock可以让等待锁的线程响应中断,而synchronized却不行,使用synchronized时,等待的线程会一直等待下去,不能够响应中断;
  • 通过Lock可以知道有没有成功获取锁,而synchronized却无法办到;
  • Lock可以提高多个线程进行读操作的效率。

在性能上来说,如果竞争资源不激烈,两者的性能是差不多的,而当竞争资源非常激烈时,此时Lock的性能要远远优于synchronized。

2. Lock接口的主要方法

Lock是java.util.concurrent.locks包中的一个接口,其主要方法有获取锁的方法:lock()、tryLock()、tryLock(long time, TimeUnit unit)和lockInterruptibly(),释放锁的方法unLock()。

(1) lock()方法

lock()方法用来获取锁,若其他锁被其他线程获取,则等待,直到获取到。所以说一个线程是使用完锁后一定要释放,否则会导致其他线程一直等待。一般会配合try-catch块进行使用,并在finally块中释放锁。

Lock lock = ...;
lock.lock();
try{
    // 处理任务
}catch(Exception ex){
    // 异常处理
}finally{
    lock.unlock(); // 释放锁
}

(2) tryLock()方法

tryLock()方法与lock()方法类似,不同在于前者有返回值,当获取锁成功时则返回true,反之返回false。也就是说这个方法无论能不能获取到锁都会立即返回。

tryLock(long time, TimeUnit unit)方法在tryLock()方法的基础上会等待指定的时间,若时间过后还没获取锁才返回false。两者用法类似。

Lock lock = ...;
if(lock.tryLock()) {
     try{
         // 处理任务
     }catch(Exception ex){
         // 异常处理         
     }finally{
         lock.unlock(); //释放锁
     } 
}else {
    // 如果不能获取锁,则直接做其他事情
}

(3) lockInterruptibly()方法

lockInterruptibly()方法获取锁后允许线程主动终止等待,即若有线程1和线程2两个线程同时使用lockInterruptibly()方法获取锁,且线程1获得了锁,而线程2在等待,那么使用thread2.interrupt()方法能够终止线程2的等待。实际上该方法不仅能打断在等待中的线程,也可以打断已获取锁的线程。该方法无返回值。

Lock lock = ...;
lock.lockInterruptibly();
try{
    // 处理任务
}catch(Exception ex){
    // 异常处理
}finally{
    lock.unlock(); // 释放锁
}

3. 实例

ReentrantLock即可重入锁,它是Lock接口的一个实现类。

public class LockTest {

    public static void main(String[] args) {
        // 可重入锁,Lock接口的一个实现类
        ReentrantLock reentrantLock = new ReentrantLock();

        Runnable runnable = new Runnable() {
            @SneakyThrows
            @Override
            public void run() {
                // 获取锁
                reentrantLock.lockInterruptibly();
                try {
                    System.out.println(Thread.currentThread().getName() + "获取了锁");
                    Thread.currentThread().sleep(5000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } finally {
                    // 最终都要释放锁
                    reentrantLock.unlock();
                    System.out.println(Thread.currentThread().getName() + "释放了锁");
                }
            }
        };

        Thread thread1 = new Thread(runnable, "Thread1");
        Thread thread2 = new Thread(runnable, "Thread2");
        thread1.start();
        thread2.start();

        try {
            // 主线程睡2s
            Thread.sleep(2000);
        } catch (Exception e) {
            e.printStackTrace();
        }

        // 打断thread2,停止等待
        thread2.interrupt();
        System.out.println("Thread2停止等待");
    }
}

其输出为,其中双斜杠后的为注释:

Thread1获取了锁  // 第0秒
Thread2停止等待  // 第2秒
$报错信息$       // 第2秒
Thread1释放了锁  // 第5秒