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

面試突擊:Synchronized有幾種用法?

開發 前端
當 synchronized 修飾靜態方法時,其作用范圍是整個程序,這個鎖對于所有調用這個鎖的對象都是互斥的。

在 Java 語言中,保證線程安全性的主要手段是加鎖,而 Java 中的鎖主要有兩種:synchronized 和 Lock,我們今天重點來看一下 synchronized 的幾種用法。

用法簡介

使用 synchronized 無需手動執行加鎖和釋放鎖的操作,我們只需要聲明 synchronized 關鍵字就可以了,JVM 層面會幫我們自動的進行加鎖和釋放鎖的操作。synchronized 可用于修飾普通方法、靜態方法和代碼塊,接下來我們分別來看。

1.修飾普通方法

synchronized 修飾普通方法的用法如下:

/**
* synchronized 修飾普通方法
*/
public synchronized void method() {
// ....
}

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

2.修飾靜態方法

synchronized 修飾靜態方法和修飾普通方法類似,它的用法如下:

/**
* synchronized 修飾靜態方法
*/
public static synchronized void staticMethod() {
// .......
}

當 synchronized 修飾靜態方法時,其作用范圍是整個程序,這個鎖對于所有調用這個鎖的對象都是互斥的。

所謂的互斥,指的是同一時間只能有一個線程能使用,其他線程只能排隊等待。

修飾普通方法 VS 修飾靜態方法

synchronized 修飾普通方法和靜態方法看似相同,但二者完全不同,對于靜態方法來說 synchronized 加鎖是全局的,也就是整個程序運行期間,所有調用這個靜態方法的對象都是互斥的,而普通方法是針對對象級別的,不同的對象對應著不同的鎖,比如以下代碼,同樣是調用兩次方法,但鎖的獲取完全不同,實現代碼如下:

import java.time.LocalDateTime;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;

public class SynchronizedUsage {
public static void main(String[] args) throws InterruptedException {
// 創建線程池同時執行任務
ExecutorService threadPool = Executors.newFixedThreadPool(10);

// 執行兩次靜態方法
threadPool.execute(() -> {
staticMethod();
});
threadPool.execute(() -> {
staticMethod();
});

// 執行兩次普通方法
threadPool.execute(() -> {
SynchronizedUsage usage = new SynchronizedUsage();
usage.method();
});
threadPool.execute(() -> {
SynchronizedUsage usage2 = new SynchronizedUsage();
usage2.method();
});
}

/**
* synchronized 修飾普通方法
* 本方法的執行需要 3s(因為有 3s 的休眠時間)
*/
public synchronized void method() {
System.out.println("普通方法執行時間:" + LocalDateTime.now());
try {
// 休眠 3s
TimeUnit.SECONDS.sleep(3);
} catch (InterruptedException e) {
e.printStackTrace();
}
}

/**
* synchronized 修飾靜態方法
* 本方法的執行需要 3s(因為有 3s 的休眠時間)
*/
public static synchronized void staticMethod() {
System.out.println("靜態方法執行時間:" + LocalDateTime.now());
try {
// 休眠 3s
TimeUnit.SECONDS.sleep(3);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

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

從上述結果可以看出,靜態方法加鎖是全局的,針對的是所有調用者;而普通方法加鎖是對象級別的,不同的對象擁有的鎖也不同。

3.修飾代碼塊

我們在日常開發中,最常用的是給代碼塊加鎖,而不是給方法加鎖,因為給方法加鎖,相當于給整個方法全部加鎖,這樣的話鎖的粒度就太大了,程序的執行性能就會受到影響,所以通常情況下,我們會使用 synchronized 給代碼塊加鎖,它的實現語法如下:

public void classMethod() throws InterruptedException {
// 前置代碼...

// 加鎖代碼
synchronized (SynchronizedUsage.class) {
// ......
}

// 后置代碼...
}

從上述代碼我們可以看出,相比于修飾方法,修飾代碼塊需要自己手動指定加鎖對象,加鎖的對象通常使用 this 或 xxx.class 這樣的形式來表示,比如以下代碼:

// 加鎖某個類
synchronized (SynchronizedUsage.class) {
// ......
}

// 加鎖當前類對象
synchronized (this) {
// ......
}

this VS class

使用 synchronized 加鎖 this 和 xxx.class 是完全不同的,當加鎖 this 時,表示用當前的對象進行加鎖,每個對象都對應了一把鎖;而當使用 xxx.class 加鎖時,表示使用某個類(而非類實例)來加鎖,它是應用程序級別的,是全局生效的,如以下代碼所示:

import java.time.LocalDateTime;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;

public class SynchronizedUsageBlock {
public static void main(String[] args) throws InterruptedException {
// 創建線程池同時執行任務
ExecutorService threadPool = Executors.newFixedThreadPool(10);

// 執行兩次 synchronized(this)
threadPool.execute(() -> {
SynchronizedUsageBlock usage = new SynchronizedUsageBlock();
usage.thisMethod();
});
threadPool.execute(() -> {
SynchronizedUsageBlock usage2 = new SynchronizedUsageBlock();
usage2.thisMethod();
});

// 執行兩次 synchronized(xxx.class)
threadPool.execute(() -> {
SynchronizedUsageBlock usage3 = new SynchronizedUsageBlock();
usage3.classMethod();
});
threadPool.execute(() -> {
SynchronizedUsageBlock usage4 = new SynchronizedUsageBlock();
usage4.classMethod();
});
}

/**
* synchronized(this) 加鎖
* 本方法的執行需要 3s(因為有 3s 的休眠時間)
*/
public void thisMethod() {
synchronized (this) {
System.out.println("synchronized(this) 加鎖:" + LocalDateTime.now());
try {
// 休眠 3s
TimeUnit.SECONDS.sleep(3);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

/**
* synchronized(xxx.class) 加鎖
* 本方法的執行需要 3s(因為有 3s 的休眠時間)
*/
public void classMethod() {
synchronized (SynchronizedUsageBlock.class) {
System.out.println("synchronized(xxx.class) 加鎖:" + LocalDateTime.now());
try {
// 休眠 3s
TimeUnit.SECONDS.sleep(3);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}

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

總結

synchronized 用 3 種用法,用它可以來修飾普通方法、靜態方法和代碼塊,其中最常用的是修飾代碼塊,而修飾代碼塊時需要指定一個加鎖對象,這個加鎖對象通常使用 this 或 xxx.class 來表示,當使用 this 時,表示使用當前對象來加鎖,而使用 class 時,表示表示使用某個類(非類對象實例)來加鎖,它是全局生效的。

責任編輯:武曉燕 來源: Java面試真題解析
相關推薦

2022-04-24 07:59:53

synchronizJVMAPI

2022-05-23 07:35:15

單例模式懶漢模式靜態內部類

2022-09-19 06:16:23

事務隔離級別Spring

2022-04-18 07:36:37

TimeUnit線程休眠

2022-04-13 14:43:05

JVM同步鎖Monitor 監視

2022-08-17 07:06:14

SpringBoot配置@Value

2022-03-07 07:33:16

線程池Java語言

2022-08-24 07:06:36

SpringSetter項目

2022-05-05 07:38:32

volatilJava并發

2022-06-06 07:35:26

MySQLInnoDBMyISAM

2024-09-27 08:57:36

2022-06-29 08:16:55

對象String字面量

2022-05-11 07:41:55

死鎖運算線程

2022-08-22 07:06:32

MyBatisSQL占位符

2022-08-31 07:04:50

Bean作用域

2022-08-03 07:04:56

GETHTTPPOST

2022-04-26 08:02:00

locktryLocklockInterr

2022-08-10 07:06:57

IoCDISpring

2022-02-08 07:02:32

進程線程操作系統

2022-08-15 07:06:50

Propertiesyml配置
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 久久一 | 成人精品啪啪欧美成 | 免费观看一级毛片 | 亚洲综合精品 | 日本天堂一区二区 | 91精品国产乱码久久蜜臀 | www.三级| 亚洲先锋影音 | www.日韩欧美 | 欧美日韩国产免费 | 九九热视频这里只有精品 | 本道综合精品 | 在线国产视频 | 国产欧美在线观看 | 1级毛片| 欧美精品一区二区三区在线播放 | 日本中文字幕在线视频 | 欧美一区二 | 国产伦一区二区三区久久 | 精品一区二区久久 | 欧美xxxⅹ性欧美大片 | 欧美成人视屏 | 久久精品 | 久草.com| 四虎成人免费电影 | 国产成人在线播放 | 国产羞羞视频在线观看 | 欧美精品一区二区在线观看 | 一道本不卡 | 中文在线а√在线8 | 99精品热视频| 亚洲精品国产成人 | 精品久久久久一区二区国产 | 日韩av在线一区 | 亚洲精品二三区 | 成人在线视频一区 | 91精品国产色综合久久 | 一区二区三区欧美 | 亚洲欧美日韩中文在线 | 中文字幕第一页在线 | 成人免费看黄 |