亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java中synchronized用法匯總

 更新時間:2022年04月13日 15:23:27   作者:Java中文社群  
使用 synchronized 無需手動執(zhí)行加鎖和釋放鎖的操作,我們只需要聲明 synchronized 關(guān)鍵字就可以了,JVM 層面會幫我們自動的進行加鎖和釋放鎖的操作,我們今天重點來看一下synchronized 的幾種用法

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

用法簡介

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

1、修飾普通方法

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

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

當(dāng) synchronized 修飾普通方法時,被修飾的方法被稱為同步方法,其作用范圍是整個方法,作用的對象是調(diào)用這個方法的對象。

2、修飾靜態(tài)方法

synchronized 修飾靜態(tài)方法和修飾普通方法類似,它的用法如下:

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

當(dāng) synchronized 修飾靜態(tài)方法時,其作用范圍是整個程序,這個鎖對于所有調(diào)用這個鎖的對象都是互斥的。

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

修飾普通方法 VS 修飾靜態(tài)方法

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

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 {
        // 創(chuàng)建線程池同時執(zhí)行任務(wù)
        ExecutorService threadPool = Executors.newFixedThreadPool(10);

        // 執(zhí)行兩次靜態(tài)方法
        threadPool.execute(() -> {
            staticMethod();
        });
        
        // 執(zhí)行兩次普通方法
            SynchronizedUsage usage = new SynchronizedUsage();
            usage.method();
            SynchronizedUsage usage2 = new SynchronizedUsage();
            usage2.method();
    }

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

     * synchronized 修飾靜態(tài)方法
    public static synchronized void staticMethod() {
        System.out.println("靜態(tài)方法執(zhí)行時間:" + LocalDateTime.now());
}

以上程序的執(zhí)行結(jié)果如下:

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

3、修飾代碼塊

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

public void classMethod() throws InterruptedException {
    // 前置代碼...
    
    // 加鎖代碼
    synchronized (SynchronizedUsage.class) {
        // ......
    }
    
    // 后置代碼...
}

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

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

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

this VS class

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

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 {
        // 創(chuàng)建線程池同時執(zhí)行任務(wù)
        ExecutorService threadPool = Executors.newFixedThreadPool(10);

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

        // 執(zhí)行兩次 synchronized(xxx.class)
            SynchronizedUsageBlock usage3 = new SynchronizedUsageBlock();
            usage3.classMethod();
            SynchronizedUsageBlock usage4 = new SynchronizedUsageBlock();
            usage4.classMethod();
    }

    /**
     * synchronized(this) 加鎖
     * 本方法的執(zhí)行需要 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) 加鎖
    public void classMethod() {
        synchronized (SynchronizedUsageBlock.class) {
            System.out.println("synchronized(xxx.class) 加鎖:" + LocalDateTime.now());
}

以上程序的執(zhí)行結(jié)果如下:

總結(jié)

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

到此這篇關(guān)于Java中synchronized用法匯總的文章就介紹到這了,更多相關(guān)java synchronized用法內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Java入門案列之猜拳小游戲

    Java入門案列之猜拳小游戲

    這篇文章主要為大家詳細介紹了Java入門案列之猜拳小游戲,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-11-11
  • mybatis批量添加,批量更新之前如何判斷是否已經(jīng)存在

    mybatis批量添加,批量更新之前如何判斷是否已經(jīng)存在

    這篇文章主要介紹了mybatis批量添加,批量更新之前如何判斷是否已經(jīng)存在,具有很好的參考價值,希望對的有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-08-08
  • Java 阻塞隊列詳解及簡單使用

    Java 阻塞隊列詳解及簡單使用

    這篇文章主要介紹了Java 阻塞隊列詳解及簡單使用的相關(guān)資料,需要的朋友可以參考下
    2017-02-02
  • Javaweb動態(tài)開發(fā)最重要的Servlet詳解

    Javaweb動態(tài)開發(fā)最重要的Servlet詳解

    動態(tài)web的核心是Servlet,由tomcat解析并執(zhí)行,本質(zhì)是Java中的一個類(面向?qū)ο螅┻@個類的功能十分強大幾乎可以完成全部功能,在Java規(guī)范中只有Servlet實現(xiàn)類實例化的對象才能被瀏覽器訪問,所以掌握Servlet具有重要意義
    2022-08-08
  • Java中的紙老虎之泛型

    Java中的紙老虎之泛型

    泛型在java中有很重要的地位,在面向?qū)ο缶幊碳案鞣N設(shè)計模式中有非常廣泛的應(yīng)用。對java的泛型特性的了解僅限于表面的淺淺一層,直到在學(xué)習(xí)設(shè)計模式時發(fā)現(xiàn)有不了解的用法,才想起詳細的記錄一下。
    2021-09-09
  • 提升網(wǎng)絡(luò)請求穩(wěn)定性HttpClient的重試機制深入理解

    提升網(wǎng)絡(luò)請求穩(wěn)定性HttpClient的重試機制深入理解

    這篇文章主要為大家介紹了提升網(wǎng)絡(luò)請求穩(wěn)定性HttpClient的重試機制深入理解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-10-10
  • 解決springboot中自定義JavaBean返回的json對象屬性名稱大寫變小寫問題

    解決springboot中自定義JavaBean返回的json對象屬性名稱大寫變小寫問題

    開發(fā)過程中發(fā)現(xiàn)查詢返回的數(shù)據(jù)出現(xiàn)自定義的JavaBean的屬性值大小寫格式出現(xiàn)問題,導(dǎo)致前端無法接受到數(shù)據(jù),目前有四種解決方法,根據(jù)大佬的經(jīng)驗之談,前兩種是最簡單便捷的,后兩種是比較通用的方法,需要的朋友可以參考下
    2023-10-10
  • Java模版引擎Freemarker

    Java模版引擎Freemarker

    FreeMarker是一個模板引擎,一個基于模板生成文本輸出的通用工具,使用純Java編寫 FreeMarker被設(shè)計用來生成HTML Web頁面,特別是基于MVC模式的應(yīng)用程序
    2016-04-04
  • Java中的StringTokenizer實現(xiàn)字符串切割詳解

    Java中的StringTokenizer實現(xiàn)字符串切割詳解

    這篇文章主要介紹了Java中的StringTokenizer實現(xiàn)字符串切割詳解,java.util工具包提供了字符串切割的工具類StringTokenizer,Spring等常見框架的字符串工具類(如Spring的StringUtils),需要的朋友可以參考下
    2024-01-01
  • Java不可變類機制淺析

    Java不可變類機制淺析

    所謂的不可變類是指這個類的實例一旦創(chuàng)建完成后,就不能改變其成員變量值。如JDK內(nèi)部自帶的很多不可變類:Interger、Long和String等。接下來通過本文給大家介紹Java不可變類機制,需要的朋友參考下
    2017-02-02

最新評論