程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> JAVA編程 >> 關於JAVA >> Java 鎖的常識總結及實例代碼

Java 鎖的常識總結及實例代碼

編輯:關於JAVA

Java 鎖的常識總結及實例代碼。本站提示廣大學習愛好者:(Java 鎖的常識總結及實例代碼)文章只能為提供參考,不一定能成為您想要的結果。以下是Java 鎖的常識總結及實例代碼正文


java中有哪些鎖

這個成績在我看了一遍<java並發編程>後盡然沒法答復,解釋本身關於鎖的概念懂得的不敷。因而再次翻看了一下書裡的內容,忽然有點翻開腦門的感到。看來確切是要進修的最好方法是要帶著成績去學,而且處理成績。

在java中鎖重要兩類:外部鎖synchronized和顯示鎖java.util.concurrent.locks.Lock。但細細想這貌似總結的也不太對。應當是由java內置的鎖和concurrent完成的一系列鎖。

為何這說,由於在java中一切都是對象,而java對每一個對象都內置了一個鎖,也能夠稱為對象鎖/外部鎖。經由過程synchronized來完成相干的鎖操作。

而由於synchronized的完成有些缺點和並發場景的龐雜性,有人開辟了一種顯式的鎖,而這些鎖都是由java.util.concurrent.locks.Lock派生出來的。固然今朝曾經內置到了JDK1.5及以後的版本中。

synchronized

起首來看看用的比擬多的synchronized,我的平常任務中年夜多用的也是它。synchronized是用於為某個代碼塊的供給鎖機制,在java的對象中會隱式的具有一個鎖,這個鎖被稱為內置鎖(intrinsic)或監督器鎖(monitor locks)。線程在進入被synchronized掩護的塊之前主動取得這個鎖,直到完成代碼後(也能夠是異常)主動釋放鎖。內置鎖是互斥的,一個鎖同時只能被一個線程持有,這也就會招致多線程下,鎖被持有後前面的線程會壅塞。正是以完成了對代碼的線程平安包管了原子性。

可重入

既然java內置鎖是互斥的並且前面的線程會招致壅塞,那末假如持有鎖的線程再次進入試圖取得這個鎖時會若何呢?好比上面的一種情形:

public class BaseClass {
  public synchronized void do() {
    System.out.println("is base");
  }

}

public class SonClass extends BaseClass {
  public synchronized void do() {
   System.out.println("is son");
   super.do();
  }

}


SonClass son = new SonClass();
son.do();

此時派生類的do辦法除會起首會持有一次鎖,然後在挪用super.do()的時刻又會再一次進入鎖並去持有,假如鎖是互斥的話此時就應當逝世鎖了。

但成果卻不是如許的,這是由於外部鎖是具有可重入的特征,也就是鎖完成了一個重入機制,援用計數治理。當線程1持有了對象的鎖a,此時會對鎖a的援用盤算加1。然後當線程1再次取得鎖a時,線程1照樣持有鎖a的那末盤算會加1。固然每次加入同步塊時會減1,直到為0時釋放鎖。

synchronized的一些特色

潤飾代碼的方法

潤飾辦法

public class BaseClass {
  public synchronized void do() {
    System.out.println("is base");
  }

}

這類就是直接對某個辦法停止加鎖,進入這個辦法塊時須要取得鎖。

潤飾代碼塊

public class BaseClass {
  private static Object lock = new Object();
  public void do() {
    synchronized (lock) {
      System.out.println("is base");
    }
  }

}

這裡就將鎖的規模削減到了辦法中的部門代碼塊,這關於鎖的靈巧性就進步了,究竟鎖的粒度掌握也是鎖的一個症結成績。

對象鎖的類型

常常看到一些代碼中對synchronized應用比擬特殊,看一下以下的代碼:

public class BaseClass {
  private static Object lock = new Object();
  public void do() {
    synchronized (lock) {
    }
  }
  
  public synchronized void doVoid() {
  }
  
  public synchronized static void doStaticVoid() {
  }
  
  public static void doStaticVoid() {
    synchronized (BaseClass.class) {
    
    }
  }  

}

這裡湧現了四種情形:潤飾代碼塊,潤飾了辦法,潤飾了靜態辦法,潤飾BaseClass的class對象。那這幾種情形會有甚麼分歧呢?

潤飾代碼塊

這類情形下我們創立了一個對象lock,在代碼中應用synchronized(lock)這類情勢,它的意思是應用lock這個對象的內置鎖。這類情形下就將鎖的掌握交給了一個對象。固然這類情形還有一種方法:

public void do() {
  synchronized (this) {
    System.out.println("is base");
  }
}

應用this的意思就是以後對象的鎖。這裡也道出了內置鎖的症結,我供給一把鎖來掩護這塊代碼,不管哪一個線程來都面臨統一把鎖咯。

潤飾對象辦法

這類直接潤飾在辦法是咱個情形?其實和潤飾代碼塊相似,只不外此時默許應用的是this,也就是以後對象的鎖。如許寫起代碼來倒也比擬簡略明白。後面說過了與潤飾代碼塊的差別重要照樣掌握粒度的差別。

潤飾靜態辦法

靜態辦法豈非有啥紛歧樣嗎?確切是紛歧樣的,此時獲得的鎖曾經不是this了,而this對象指向的class,也就是類鎖。由於Java中的類信息會加載到辦法常量區,全局是獨一的。這其實就供給了一種全局的鎖。

潤飾類的Class對象

這類情形其實和修正靜態辦法時比擬相似,只不外照樣一個事理這類方法可以供給更靈巧的掌握粒度。

小結

經由過程這幾種情形的剖析與懂得,其實可以看內置鎖的重要焦點理念就是為一塊代碼供給一個可以用於互斥的鎖,起到相似於開關的功效。

java中對內置鎖也供給了一些完成,重要的特色就是java都是對象,而每一個對象都有鎖,所以可以依據情形選擇用甚麼樣的鎖。

java.util.concurrent.locks.Lock

後面看了synchronized,年夜部門的情形下差不多就夠啦,然則如今體系在並發編程中龐雜性是愈來愈高,所以老是有很多場景synchronized處置起來會比擬費力。或許像<java並發編程>中說的那樣,concurrent中的lock是對外部鎖的一種彌補,供給了更多的一些高等特征。

java.util.concurrent.locks.Lock簡略剖析

這個接口籠統了鎖的重要操作,也是以讓從Lock派生的鎖具有了這些根本的特征:無前提的、可輪循的、准時的、可中止的。並且加鎖與解鎖的操作都是顯式停止。上面是它的代碼:

public interface Lock {
  void lock();
  void lockInterruptibly() throws InterruptedException;
  boolean tryLock();
  boolean tryLock(long time, TimeUnit unit) throws InterruptedException;
  void unlock();
  Condition newCondition();
}

ReentrantLock

ReentrantLock就是可重入鎖,連名字都這麼顯式。ReentrantLock供給了和synchronized相似的語義,然則ReentrantLock必需顯式的挪用,好比:

public class BaseClass {
  private Lock lock = new ReentrantLock();

  public void do() {
    lock.lock();
    try {
    //....
    } finally {
     lock.unlock();
    }
    
  }

}

這類方法關於代碼浏覽來講照樣比擬清晰的,只不外有個成績,就是假如忘了加try finally或忘 了寫lock.unlock()的話招致鎖沒釋放,很有能夠招致一些逝世鎖的情形,synchronized就沒有這個風險。

trylock

ReentrantLock是完成Lock接口,所以天然就具有它的那些特征,個中就有trylock。trylock就是測驗考試獲得鎖,假如鎖曾經被其他線程占用那末立刻前往false,假如沒有那末應當占用它並前往true,表現拿到鎖啦。

另外一個trylock辦法裡帶了參數,這個辦法的感化是指定一個時光,表現在這個時光內一向測驗考試去取得鎖,假如到時光還沒有拿到就廢棄。

由於trylock對鎖其實不是一向壅塞期待的,所以可以更多的躲避逝世鎖的產生。

lockInterruptibly

lockInterruptibly是在線程獲得鎖時優先呼應中止,假如檢測到中止拋出中止異常由下層代碼行止理。這類情形下就為一種輪循的鎖供給了加入機制。為了更好懂得可中止的鎖操作,寫了一個demo來懂得。

package com.test;

import java.util.Date;
import java.util.concurrent.locks.ReentrantLock;

public class TestLockInterruptibly {
 static ReentrantLock lock = new ReentrantLock();
 
 public static void main(String[] args) {
 Thread thread1 = new Thread(new Runnable() {
  
  @Override
  public void run() {
  try {
   doPrint("thread 1 get lock.");
   do123();
   doPrint("thread 1 end.");
   
  } catch (InterruptedException e) {
   doPrint("thread 1 is interrupted.");
  }
  }
 });

 Thread thread2 = new Thread(new Runnable() {
  
  @Override
  public void run() {
  try {
   doPrint("thread 2 get lock.");
   do123();
   doPrint("thread 2 end.");   
  } catch (InterruptedException e) {
   doPrint("thread 2 is interrupted.");
  }
  }
 });
 
 thread1.setName("thread1");
 thread2.setName("thread2");
 thread1.start();
 try {
  Thread.sleep(100);//期待一會使得thread1會在thread2後面履行  
 } catch (InterruptedException e) {
  e.printStackTrace();
 }
 thread2.start();
 }
 
 
 private static void do123() throws InterruptedException {
 lock.lockInterruptibly();
 doPrint(Thread.currentThread().getName() + " is locked.");
 try {
  doPrint(Thread.currentThread().getName() + " doSoming1....");
  Thread.sleep(5000);//期待幾秒便利檢查線程的前後次序
  doPrint(Thread.currentThread().getName() + " doSoming2....");

  doPrint(Thread.currentThread().getName() + " is finished.");
 } finally {
  lock.unlock();
 }
 }
 
 private static void doPrint(String text) {
 System.out.println((new Date()).toLocaleString() + " : " + text);
 }
}

下面代碼中有兩個線程,thread1比thread2更早啟動,為了能看到拿鎖的進程將上鎖的代碼sleep了5秒鐘,如許便可以感觸感染到前後兩個線程進入獲得鎖的進程。終究下面的代碼運轉成果以下:

2016-9-28 15:12:56 : thread 1 get lock.
2016-9-28 15:12:56 : thread1 is locked.
2016-9-28 15:12:56 : thread1 doSoming1....
2016-9-28 15:12:56 : thread 2 get lock.
2016-9-28 15:13:01 : thread1 doSoming2....
2016-9-28 15:13:01 : thread1 is finished.
2016-9-28 15:13:01 : thread1 is unloaded.
2016-9-28 15:13:01 : thread2 is locked.
2016-9-28 15:13:01 : thread2 doSoming1....
2016-9-28 15:13:01 : thread 1 end.
2016-9-28 15:13:06 : thread2 doSoming2....
2016-9-28 15:13:06 : thread2 is finished.
2016-9-28 15:13:06 : thread2 is unloaded.
2016-9-28 15:13:06 : thread 2 end.

可以看到,thread1先取得鎖,一會thread2也來拿鎖,但這個時刻thread1曾經占用了,所以thread2一向到thread1釋放了鎖後才拿到鎖。

**這段代碼解釋lockInterruptibly前面來獲得鎖的線程須要期待後面的鎖釋放了能力取得鎖。**但這裡還沒有表現出可中止的特色,為此增長一些代碼:

thread2.start();
try {
 Thread.sleep(1000);  
} catch (InterruptedException e) {
 e.printStackTrace();
} 
//1秒後把線程2中止
thread2.interrupt();

在thread2啟動後挪用一下thread2的中止辦法,好吧,先跑一下代碼看看成果:

2016-9-28 15:16:46 : thread 1 get lock.
2016-9-28 15:16:46 : thread1 is locked.
2016-9-28 15:16:46 : thread1 doSoming1....
2016-9-28 15:16:46 : thread 2 get lock.
2016-9-28 15:16:47 : thread 2 is interrupted. <--直接就呼應了線程中止
2016-9-28 15:16:51 : thread1 doSoming2....
2016-9-28 15:16:51 : thread1 is finished.
2016-9-28 15:16:51 : thread1 is unloaded.
2016-9-28 15:16:51 : thread 1 end.
和後面的代碼比擬可以發明,thread2正在期待thread1釋放鎖,然則這時候thread2本身中止了,thread2前面的代碼則不會再持續履行。

ReadWriteLock

望文生義就是讀寫鎖,這類讀-寫鎖的運用場景可以如許懂得,好比一波數據年夜部門時刻都是供給讀取的,而只要比擬大批的寫操作,那末假如用互斥鎖的話就會招致線程間的鎖競爭。假如關於讀取的時刻年夜家都可以讀,一旦要寫入的時刻就再將某個資本鎖住。如許的變更就很好的處理了這個成績,使的讀操作可以進步讀的機能,又不會影響寫的操作。

一個資本可以被多個讀者拜訪,或許被一個寫者拜訪,二者不克不及同時停止。

這是讀寫鎖的籠統接口,界說一個讀鎖和一個寫鎖。

public interface ReadWriteLock {
  /**
   * Returns the lock used for reading.
   *
   * @return the lock used for reading
   */
  Lock readLock();

  /**
   * Returns the lock used for writing.
   *
   * @return the lock used for writing
   */
  Lock writeLock();
}

在JDK裡有個ReentrantReadWriteLock完成,就是可重入的讀-寫鎖。ReentrantReadWriteLock可以結構為公正的或許非公正的兩品種型。假如在結構時不顯式指定章會默許的創立非公正鎖。在非公正鎖的形式下,線程拜訪的次序是不肯定的,就是可以闖入;可以由寫者升級為讀者,然則讀者不克不及進級為寫者。

假如是公正鎖形式,那末選擇權交給期待時光最長的線程,假如一個讀線程取得鎖,此時一個寫線程要求寫入鎖,那末就不再吸收讀鎖的獲得,直到寫入操作完成。

簡略的代碼剖析 在ReentrantReadWriteLock裡其實保護的是一個sync的鎖,只是看起來語義上像是一個讀鎖和寫鎖。看一下它的結構函數:

public ReentrantReadWriteLock(boolean fair) {
  sync = fair ? new FairSync() : new NonfairSync();
  readerLock = new ReadLock(this);
  writerLock = new WriteLock(this);
}

//讀鎖的結構函數
protected ReadLock(ReentrantReadWriteLock lock) {
  sync = lock.sync;
}
//寫鎖的結構函數
protected WriteLock(ReentrantReadWriteLock lock) {
  sync = lock.sync;
}

可以看到現實上讀/寫鎖在結構時都是援用的ReentrantReadWriteLock的sync鎖對象。而這個Sync類是ReentrantReadWriteLock的一個外部類。總之讀/寫鎖都是經由過程Sync來完成的。它是若何來協作這二者關系呢?

//讀鎖的加鎖辦法
public void lock() {
  sync.acquireShared(1);
}

//寫鎖的加鎖辦法
public void lock() {
  sync.acquire(1);
}

差別重要是讀鎖取得的是同享鎖,而寫鎖獲得的是獨有鎖。這裡有個點可以提一下,就是ReentrantReadWriteLock為了包管可重入性,同享鎖和獨有鎖都必需支撐持有計數和重入數。而ReentrantLock是應用state來存儲的,而state只能存一個整形值,為了兼容兩個鎖的成績,所以將其劃分了高16位和低16位分離存同享鎖的線程數目或獨有鎖的線程數目或許重入計數。

其他

寫了一年夜篇感到要寫下去篇幅太長了,還有一些比擬有效的鎖:

CountDownLatch

就是設置一個同時持有的計數器,而挪用者挪用CountDownLatch的await辦法時假如以後的計數器不為0就會壅塞,挪用CountDownLatch的release辦法可以削減計數,直到計數為0時挪用了await的挪用者會消除壅塞。

Semaphone

旌旗燈號量是一種經由過程受權允許的情勢,好比設置100個允許證,如許便可以同時有100個線程同時持有鎖,假如跨越這個量後就會前往掉敗。

感激浏覽此文,願望能贊助到年夜家,感謝年夜家對本站的支撐!

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved