
**Java synchronized 關鍵字** 可以將一個代碼塊或一個方法標記為同步代碼塊。同步代碼塊是指同一時間只能有一個線程執行的代碼,并且執行該代碼的線程持有同步鎖。`synchronized`關鍵字可以作用于
* 一個代碼塊
* 一種方法
當一個方法或代碼塊被聲明為**synchronized**時,如果一個線程正在執行該**synchronized** 方法或代碼塊,其他線程會被阻塞,直到持有同步鎖的線程釋放。根據鎖定的范圍可以分為
* 類級別的鎖可以防止多個線程在運行時同時進入**該類所有實例化對象**的 synchronized代碼塊中。
* 對象級別的鎖可以防止多個線程在運行時同時進入**當前(或某一個)實例化對象**的 synchronized代碼塊中。
## 1. 對象級別的同步鎖
對象級別的同步鎖:當我們想要在多線程環境下同步執行一個非靜態方法或非靜態代碼塊時,在類的方法或代碼塊加上**synchronized**關鍵字,可以保證對象實例級別數據的線程安全。(比較后文的類級別的同步鎖,回頭來理解這句話)
對象級別的加鎖的代碼如下,如:在方法上加鎖,鎖對象為當前類的實例化對象
~~~
public class DemoClass{
public synchronized void demoMethod(){}
}
~~~
如:為代碼塊加鎖,鎖對象為this對象
~~~
public class DemoClass{
public void demoMethod(){
synchronized (this){
//同步代碼塊
}
}
}
~~~
如:為代碼塊加鎖,鎖對象為我們創建的任意一個對象。不要使用非final的成員變量作為同步鎖對象,因為非final成員變量可以被重新賦值,導致不同的線程使用不同的對象作為鎖,達不到同步鎖定的效果。
~~~
public class DemoClass{
//注意這里的關鍵字final非常重要,看說明
private final Object lock = new Object();
public void demoMethod(){
synchronized (lock){
//同步代碼塊
}
}
}
~~~
## 2. 類級別的同步鎖
類級別的鎖可以防止多個線程在運行時進入該類所有實例化對象的 "synchronized塊中。也就是說如果運行時有100個`DemoClass`的實例,那么每次只有一個線程能夠在任何一個實例中執行`demoMethod()`,所有其他實例的所有其他線程都被鎖定。
為了保障靜態數據線程安全,應該使用類級別的鎖定。我們知道static關鍵字將方法的數據關聯到類的級別上,所以在靜態方法上使用鎖。
靜態方法加鎖,對該類所有的實例化對象生效
~~~
public class DemoClass{
//靜態方法加鎖,對該類所有的實例化對象生效
public synchronized static void demoMethod(){
}
}
~~~
獲取 .class類的引用,類級別的鎖
~~~
public class DemoClass{
public void demoMethod(){
//獲取 .class類的引用,類級別的鎖,對該類所有的實例化對象生效
synchronized (DemoClass.class){
//同步代碼塊
}
}
}
~~~
使用靜態對象的鎖,類級別的鎖
~~~
public class DemoClass{
//靜態對象,類級別,注意這里的關鍵字final非常重要
private final static Object lock = new Object();
public void demoMethod(){
//使用靜態對象的鎖,類級別鎖,對該類所有的實例化對象生效
synchronized (lock){
//同步代碼塊
}
}
}
~~~
## 3. 總結
1. Java中的同步機制保證了兩個或多個線程無法同時執行一個需要相同同步鎖的方法。
2. "synchronized "關鍵字只能用于方法和代碼塊。這些方法或代碼塊可以是*靜態*或*非靜態*的。
3. 當一個線程進入`synchronized`方法或代碼塊時,它就會獲得一個鎖,當它離開同步方法或代碼塊時,它就會釋放這個鎖。如果線程執行過程出現任何錯誤或異常,鎖也會被釋放。
4. 使用"synchronized "關鍵字持有的鎖在本質上是可重入的,這意味著如果一個同步方法調用另一個**使用相同鎖**的同步方法,那么持有鎖的當前線程可以進入該方法而無需再次獲得鎖。
5. 如果同步塊中使用的對象為空,Java synchronized 將拋出NullPointerException
6. 使用`synchronized`同步方法會給你的應用程序帶來性能成本。因此,盡量在絕對需要的情況下才使用同步。另外優先考慮使用同步代碼塊,并且只同步代碼的關鍵部分。
7. 靜態同步方法和非靜態同步方法有可能同時或并發運行,因為它們使用的是不同的鎖。
8. 根據Java語言規范,你不能在構造函數中使用`synchronized`關鍵字。這是不合法的,會導致編譯錯誤。
9. 不要使用非final的成員變量作為同步鎖對象,因為非final成員變量可以被重新賦值,導致不同的線程使用不同的對象作為鎖,達不到同步鎖定的效果。
10. 不要使用字符串字面量作為鎖對象,如:`String a = "1";`,因為它們可能會被應用程序中的其他地方引用,并可能導致死鎖。用`new`關鍵字創建的字符串對象可以安全使用。
- 線程
- 1.進程和線程-鎖與信號量
- 2.Thread類線程狀態轉換
- 2.并發與并行-同步與異步
- 4.線程池
- 5.對象級別與類級別的同步鎖
- 6.創建線程的四種方式
- 7.臨界區-阻塞-活鎖-死鎖
- 2.JMM多線程模型
- JUC
- BlockingQueue
- ArrayBlockingQueue
- DelayQueue
- LinkedBlockingQueue
- PriorityBlockingQueue
- SynchronousQueue
- BlockingDeque
- ConcurrentHashMap
- CountDownLatch
- CyclicBarrier
- Exchanger
- AtomicInteger
- Lock
- Condition
- ReentrantLock讀寫鎖
- StampedLock
- Semaphore