Lock

  1. 从JDK5.0开始,Java提供了更强大的线程同步机制----通过显示定义同步锁对象来实现同步。同步锁使用Lock对象充当
  2. java.util.concurrent.locks.Lock接口是控制多个线程对共享资源进行访问的工具,锁提供了对共享资源的独占访问,每次只能有一个线程对Lock对象加锁,线程开始访问共享资源前应先获得Lock对象
  3. ReentrantLock类实现了Lock,它拥有与synchronized相同的并发性和内存语义,在实现线程安全的控制中,比较常用的是ReentrantLock,可以显式加锁,释放锁

synchronized和Lock的对比

  1. Lock是是显式锁(手动开启和关闭锁);synchronized是隐式锁,出了作用域自动释放
  2. Lock只有代码块锁;synchronized有代码块锁和方法锁
  3. 使用Lock锁,JVM将花费较少的时间来调度线程,性能更好。并且具有更好的扩展性(提供了更多的子类)
  4. 优先使用顺序(从高到低):
    1. Lock
    2. 同步代码块(已经进入了方法体,分配了相应资源)
    3. 同步方法(在方法体之外)

代码示例

import java.util.concurrent.locks.ReentrantLock;
public class TestLock {
    public static void main(String[] args) {
        Ticket ticket = new Ticket();
        //多个线程操作同一对象
        new Thread(ticket).start();
        new Thread(ticket).start();
        new Thread(ticket).start();
    }
}
//票
class Ticket implements Runnable {
    int ticketNum=10;   //票数
    //定义lock锁
    private final ReentrantLock lock = new ReentrantLock();
    @Override
    public void run() {
        while (true) {
            lock.lock();    //加锁
            try {
                //保证线程安全的代码
                if(ticketNum>0){
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    System.out.println(ticketNum--);
                }else{
                    break;
                }
            }finally {
                //如果同步代码块有异常,要将unlock()写入finally语句块
                lock.unlock();  //别忘记关闭锁哦~~
            }
        }
    }
}