Java多線程專題之Lock鎖的使用
Lock基本使用
Lock它是java.util.concurrent.locks下的一個(gè)接口,它也是用來處理線程同步問題的。
public interface Lock {
void lock();
void lockInterruptibly() throws InterruptedException;
boolean tryLock();
boolean tryLock(long time, TimeUnit unit) throws InterruptedException;
void unlock();
Condition newCondition();
}
- lock()獲取鎖,如果鎖不可用,則當(dāng)前線程將被禁用以用于線程調(diào)度目的并處于休眠狀態(tài),直到獲得鎖為止。
- lockInterruptibly()除非當(dāng)前線程被中斷,否則獲取鎖。如果可用,則獲取鎖并立即返回。如果鎖不可用,則當(dāng)前線程將被禁用以用于線程調(diào)度目的并處于休眠狀態(tài),直到鎖被當(dāng)前線程獲取或者其它線程中斷當(dāng)前線程。
- tryLock()這種用法確保鎖在獲得時(shí)解鎖,并且在未獲得鎖時(shí)不嘗試解鎖。返回布爾類型(true/false)。
- tryLock(long time, TimeUnit unit)如果在給定的等待時(shí)間內(nèi)空閑并且當(dāng)前線程沒有被中斷,則獲取鎖,如果鎖可用,則此方法立即返回值true 。如果鎖不可用,則當(dāng)前線程將被禁用以用于線程調(diào)度目的并處于休眠狀態(tài),直到當(dāng)前線程獲取鎖,或被中斷,或指定的等待時(shí)間已到。
- unlock()釋放鎖。
- newCondition()返回綁定到此Lock實(shí)例的新Condition實(shí)例。
因?yàn)樗皇墙涌?,所以我們需要找到它的?shí)現(xiàn)類,下面重點(diǎn)給大家介紹ReentrantLock,它也是我們工作中常用的。
ReentrantLock
它是一種可重入互斥Lock ,其基本行為和語義與使用synchronized方法和語句訪問的隱式監(jiān)視器鎖相同,但具有擴(kuò)展功能。
說了這么多,還沒帶大家體驗(yàn)一把,下面我們就是來個(gè)例子試一下。
public class LockTest {
private static Lock lock = new ReentrantLock();
private static int count = 0;
private static void add() {
try {
//加鎖
lock.lock();
count++;
//重入鎖
reduce();
} finally {
// 記得釋放鎖
lock.unlock();
System.out.println(count);
}
}
private static void reduce(){
// 加鎖
lock.lock();
count--;
// 釋放鎖
lock.unlock();
}
public static void main(String[] args) throws InterruptedException {
for (int i = 0; i < 100; i++) {
new Thread(LockTest::add).start();
}
}
}
我們發(fā)現(xiàn)無論執(zhí)行多少次結(jié)果都是和預(yù)期的一樣,都是0,我們?cè)倏聪聇ryLock,直接改寫reduce:
private static void reduce1(){
if(lock.tryLock()) {
try {
count--;
}finally {
// 釋放鎖
lock.unlock();
}
}else {
System.out.println("no");
}
}
通過上面的使用,我們來和Synchronized做一個(gè)對(duì)比。
- 表面上一個(gè)是關(guān)鍵字,一個(gè)是類。
- 使用上,Synchronized隱式,而Lock需要顯示,對(duì)代碼要求比較高,如果忘了釋放鎖可能會(huì)導(dǎo)致死鎖。
- Lock提供了tryLock方法,使得程序更加靈活。
- 代碼表現(xiàn)上Lock更加的靈活,可以在不同的方法中執(zhí)行。
結(jié)束語
本節(jié)主要講了它的基本使用,大家可以舉一反三,試試什么條件下會(huì)導(dǎo)致死鎖。