成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Synchronized 加鎖 This 和Class 的區別!

開發 后端
synchronized 是 Java 語言中處理并發問題的一種常用手段,它也被我們親切的稱之為“Java 內置鎖”,由此可見其地位之高。然而 synchronized 卻有著多種用法,當它修飾不同對象時,其意義也是不同的,下面我們一起來看。

[[414654]]

本文轉載自微信公眾號「Java中文社群」,作者磊哥。轉載本文請聯系Java中文社群公眾號。

synchronized 是 Java 語言中處理并發問題的一種常用手段,它也被我們親切的稱之為“Java 內置鎖”,由此可見其地位之高。然而 synchronized 卻有著多種用法,當它修飾不同對象時,其意義也是不同的,下面我們一起來看。

synchronized 用法

synchronized 可以用來修飾普通方法、靜態方法和代碼塊。

① 修飾普通方法

  1. /** 
  2.  * synchronized 修飾普通方法 
  3.  */ 
  4. public synchronized void method() { 
  5.     // ....... 

當 synchronized 修飾普通方法時,被修飾的方法被稱為同步方法,其作用范圍是整個方法,作用的對象是調用這個方法的對象。

② 修飾靜態方法

  1. /** 
  2.  * synchronized 修飾靜態方法 
  3.  */ 
  4. public static synchronized void staticMethod() { 
  5.     // ....... 

當 synchronized 修飾靜態的方法時,其作用的范圍是整個方法,作用對象是調用這個類的所有對象。

③ 修飾代碼塊

為了減少鎖的粒度,我們可以選擇在一個方法中的某個部分使用 synchronized 來修飾(一段代碼塊),從而實現對一個方法中的部分代碼進行加鎖,實現代碼如下:

  1. public void classMethod() throws InterruptedException { 
  2.     // 前置代碼... 
  3.      
  4.     // 加鎖代碼 
  5.     synchronized (SynchronizedExample.class) { 
  6.         // ...... 
  7.     } 
  8.      
  9.     // 后置代碼... 

以上代碼在執行時,被修飾的代碼塊稱為同步語句塊,其作用范圍是大括號“{}”括起來的代碼塊,作用的對象是調用這個代碼塊的對象。

但以上代碼,除了可以加鎖 class 之外,還可以加鎖 this,具體示例如下:

  1. public void classMethod() throws InterruptedException { 
  2.     // 前置處理代碼... 
  3.     synchronized (this) { 
  4.         // ...... 
  5.     } 
  6.     // 后置處理代碼... 

那問題來了,使用 synchronized 加鎖 this 和 class 的區別是什么?不都是加鎖同一個類嗎?

答案還真不是,加鎖 this 和 class 區別還是很大的。下面我們通過以下 4 個示例,來看二者之間的區別。

1.加鎖 class 共享一個類實例

首先,我們創建 5 個線程,調用同一個對象下 synchronized 加鎖的 class 代碼,具體示例如下:

  1. import java.util.Date
  2. import java.util.concurrent.TimeUnit; 
  3.  
  4. public class SynchronizedExample { 
  5.  
  6.     public static void main(String[] args) { 
  7.         // 創建當前類實例 
  8.         final SynchronizedExample example = new SynchronizedExample(); 
  9.         // 創建 5 個線程執行任務 
  10.         for (int i = 0; i < 5; i++) { 
  11.             new Thread(new Runnable() { 
  12.                 @Override 
  13.                 public void run() { 
  14.                     try { 
  15.                         // 調用 synchronized 修飾的 class 方法 
  16.                         example.classMethod(); 
  17.                     } catch (InterruptedException e) { 
  18.                         e.printStackTrace(); 
  19.                     } 
  20.                 } 
  21.             }).start(); 
  22.         } 
  23.     } 
  24.  
  25.     /** 
  26.      * synchronized 修飾的 class 方法 
  27.      * @throws InterruptedException 
  28.      */ 
  29.     public void classMethod() throws InterruptedException { 
  30.         synchronized (SynchronizedExample.class) { 
  31.             System.out.println(String.format("當前執行線程:%s,執行時間:%s"
  32.                     Thread.currentThread().getName(), new Date())); 
  33.             TimeUnit.SECONDS.sleep(1); 
  34.         } 
  35.     } 

以上程序的執行結果如下:

從上述結果可以看出,這 5 個線程共享的是同一把鎖。

2.加鎖 class 創建多個實例

接下來,我們創建 5 個線程,調用不同對象下 synchronized 加鎖的 class 代碼,具體示例如下:

  1. import java.util.Date
  2. import java.util.concurrent.TimeUnit; 
  3.  
  4. public class SynchronizedExample { 
  5.  
  6.     public static void main(String[] args) { 
  7.         // 創建 5 個線程執行任務 
  8.         for (int i = 0; i < 5; i++) { 
  9.             new Thread(new Runnable() { 
  10.                 @Override 
  11.                 public void run() { 
  12.                     try { 
  13.                         // 創建類實例 
  14.                         SynchronizedExample example = new SynchronizedExample(); 
  15.                         // 調用 synchronized 修飾的 class 方法 
  16.                         example.classMethod(); 
  17.                     } catch (InterruptedException e) { 
  18.                         e.printStackTrace(); 
  19.                     } 
  20.                 } 
  21.             }).start(); 
  22.         } 
  23.     } 
  24.      
  25.     /** 
  26.      * synchronized 修飾的 class 方法 
  27.      * @throws InterruptedException 
  28.      */ 
  29.     public void classMethod() throws InterruptedException { 
  30.         synchronized (SynchronizedExample.class) { 
  31.             System.out.println(String.format("當前執行線程:%s,執行時間:%s"
  32.                     Thread.currentThread().getName(), new Date())); 
  33.             TimeUnit.SECONDS.sleep(1); 
  34.         } 
  35.     } 

以上程序的執行結果如下:

從上述結果可以看出,雖然是不同的對象,但它們使用的仍然是同一把鎖。

3.加鎖 this 共享一個類實例

接下來,我們創建 5 個線程,調用 synchronized 加鎖 this 的示例。首先我們這 5 個線程調用同一個對象的加鎖方法,示例代碼如下:

  1. import java.util.Date
  2. import java.util.concurrent.TimeUnit; 
  3.  
  4. public class SynchronizedExample { 
  5.  
  6.     public static void main(String[] args) { 
  7.         // 創建當前類實例 
  8.         final SynchronizedExample example = new SynchronizedExample(); 
  9.         // 創建 5 個線程執行任務 
  10.         for (int i = 0; i < 5; i++) { 
  11.             new Thread(new Runnable() { 
  12.                 @Override 
  13.                 public void run() { 
  14.                     try { 
  15.                         // 調用 synchronized 修飾的 this 方法 
  16.                         example.thisMethod(); 
  17.                     } catch (InterruptedException e) { 
  18.                         e.printStackTrace(); 
  19.                     } 
  20.                 } 
  21.             }).start(); 
  22.         } 
  23.     } 
  24.      
  25.     /** 
  26.      * synchronized 修飾的 this 方法 
  27.      * @throws InterruptedException 
  28.      */ 
  29.     public void thisMethod() throws InterruptedException { 
  30.         synchronized (this) { 
  31.             System.out.println(String.format("當前執行線程:%s,執行時間:%s"
  32.                     Thread.currentThread().getName(), new Date())); 
  33.             TimeUnit.SECONDS.sleep(1); 
  34.         } 
  35.     } 

以上程序的執行結果如下:

從上述結果可以看出,以上線程使用的都是同一把鎖。

4.加鎖 this 創建多個類實例

最后一個示例最為特殊,我們使用 synchronized 加鎖 this,讓這 5 個線程調用各自創建對象的方法,具體示例如下:

  1. import java.util.Date
  2. import java.util.concurrent.TimeUnit; 
  3.  
  4. public class SynchronizedExample { 
  5.  
  6.     public static void main(String[] args) { 
  7.         // 創建 5 個線程執行任務 
  8.         for (int i = 0; i < 5; i++) { 
  9.             new Thread(new Runnable() { 
  10.                 @Override 
  11.                 public void run() { 
  12.                     try { 
  13.                         // 創建(多個)類實例 
  14.                         SynchronizedExample example = new SynchronizedExample(); 
  15.                         // 調用 synchronized 修飾的 this 方法 
  16.                         example.thisMethod(); 
  17.                     } catch (InterruptedException e) { 
  18.                         e.printStackTrace(); 
  19.                     } 
  20.                 } 
  21.             }).start(); 
  22.         } 
  23.     } 
  24.      
  25.     /** 
  26.      * synchronized 修飾的 this 方法 
  27.      * @throws InterruptedException 
  28.      */ 
  29.     public void thisMethod() throws InterruptedException { 
  30.         synchronized (this) { 
  31.             System.out.println(String.format("當前執行線程:%s,執行時間:%s"
  32.                     Thread.currentThread().getName(), new Date())); 
  33.             TimeUnit.SECONDS.sleep(1); 
  34.         } 
  35.     } 

以上程序的執行結果如下:

從上述結果可以看出,當使用 synchronized 加鎖 this 時,如果線程調用的不是同一個對象,那么這些線程之間使用的鎖都是自己獨立的鎖,這個結果就和 synchronized 加鎖 class 的結果完全不同了。

總結

通過以上 4 個示例我們可以得出結論,當使用 synchronized 加鎖 class 時,無論共享一個對象還是創建多個對象,它們用的都是同一把鎖,而使用 synchronized 加鎖 this 時,只有同一個對象會使用同一把鎖,不同對象之間的鎖是不同的。

 

責任編輯:武曉燕 來源: Java中文社群
相關推薦

2021-08-24 10:25:19

thisclassJava

2024-02-26 07:36:09

lockJava語言

2022-04-24 07:59:53

synchronizJVMAPI

2023-02-01 07:15:16

2020-03-10 10:25:38

volatileJava編程語言

2022-11-09 10:46:18

AQS加鎖機制

2021-02-01 14:10:16

JavaClass.forNaClassLoader

2021-08-31 06:37:35

Java 語言 Java 基礎

2020-09-23 09:08:05

typescript

2009-10-10 14:40:03

C++中struct

2010-08-23 14:16:17

DIVclassid

2010-08-30 10:52:39

CSSclassid

2021-01-14 08:58:12

Synchronize鎖操作

2010-08-23 10:57:14

CSSclassid

2009-08-27 15:48:40

interfaceabstract cl

2009-08-27 16:22:58

interfaceabstract cl

2017-12-06 16:28:48

Synchronize實現原理

2021-03-04 08:26:17

synchronizeReentrantLojava

2025-01-02 12:59:55

Python面向對象編程type

2020-07-07 07:47:07

Java無鎖技術
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 一区二区三区视频播放 | 午夜天堂精品久久久久 | 97高清国语自产拍 | av黄色在线 | 欧美成人不卡 | 欧美精品成人一区二区三区四区 | 99精品久久 | 久久国产亚洲 | 久久99国产精一区二区三区 | av片免费 | 亚洲精品字幕 | 永久网站| av一区二区三区四区 | 91久久精品国产免费一区 | 人妖videosex高潮另类 | 成人免费黄视频 | 一级毛片视频 | 国产成人免费视频网站视频社区 | 91av免费版 | 在线观看www视频 | av电影一区二区 | 国产精品一区二区欧美黑人喷潮水 | 亚洲精品一区在线 | www.三级| 亚洲欧洲激情 | 九九在线视频 | 国产精品久久久久久一区二区三区 | 亚洲超碰在线观看 | 久久精品免费 | 免费在线观看黄视频 | 91久久精品一区 | 成人在线看片 | 在线观看第一区 | 久久专区| 国产激情一区二区三区 | 成人欧美一区二区三区黑人孕妇 | 亚洲第一网站 | 黄色av免费 | 成人在线免费视频 | 伊人网综合在线观看 | 国产免费xxx |