1. 程式人生 > 其它 >Java基礎——多執行緒(Lock鎖)

Java基礎——多執行緒(Lock鎖)

一、Lock鎖

雖然我們可以理解同步程式碼塊和同步方法的鎖物件問題,但是我們並沒有直接看到在哪裡上了鎖,在哪裡釋放了鎖,為了更清晰的表達如何加鎖和釋放鎖,JDK5以後提供了一個新的鎖物件Lock

Lock實現提供比使用synchronized方法和語句更廣泛的鎖定操作

二、Lock中提供了獲得鎖和釋放鎖的方法

1.void lock():獲得鎖

2.void unlock();釋放鎖

Lock由於是介面,不能直接例項化,這裡採用它的實現類ReentrantLock來例項化

三、ReentrantLock的構造方法

ReentrantLock():建立一個ReentrantLock的例項

例如:之前的賣票案例就可以這樣寫

public class SellTicket implements Runnable {
  //定義總張數
  private int tickets = 100;
  //定義Lock鎖,要用它的實現類完成
  Lock lock=new ReentrantLock();
  @Override
  public void run() {
      while (true) {
          //判斷車票是否大於0
          //加鎖
          lock.lock();
          if (tickets > 0) {
              System.out.println(Thread.currentThread().getName() + "賣出第" + tickets + "張車票");
              tickets--;
              //模仿出票
              try {
                  sleep(100);
              } catch (InterruptedException e) {
                  e.printStackTrace();
              }          
              }
          //釋放鎖
          lock.unlock();            
      }
  }
}

為了避免出現有時候出現異常不能及時釋放鎖的情況,我們用try...finally來操作一下

package Demo041901;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
import static java.lang.Thread.sleep;
public class SellTicket implements Runnable {
  //定義總張數
  private int tickets = 100;
  //定義Lock鎖,要用它的實現類完成
  Lock lock=new ReentrantLock();
  @Override
  public void run() {
      while (true) {
          //判斷車票是否大於0
          try {
              //加鎖
              lock.lock();
              if (tickets > 0) {
                  System.out.println(Thread.currentThread().getName() + "賣出第" + tickets + "張車票");
                  tickets--;
                  //模仿出票
                  try {
                      sleep(100);
                  } catch (InterruptedException e) {
                      e.printStackTrace();
                  }
              }else {
                  break;
              }
          }finally {
              //釋放鎖
              lock.unlock();
          }
      }
  }
}