synchronized

来源:http://www.cnblogs.com/Chenjiabing/archive/2017/06/21/7057635.html
-Advertisement-
Play Games

synchronized 前言 相信大家都聽說過線程安全問題,在學習操作系統的時候有一個知識點是臨界資源,簡單的說就是一次只能讓一個進程操作的資源,但是我們在使用多線程的時候是併發操作的,並不能控制同時只對一個資源的訪問和修改,想要控制那麼有幾種操作,今天我們就來講講第一種方法:線程同步塊或者線程同 ...


synchronized

前言

相信大家都聽說過線程安全問題,在學習操作系統的時候有一個知識點是臨界資源,簡單的說就是一次只能讓一個進程操作的資源,但是我們在使用多線程的時候是併發操作的,並不能控制同時只對一個資源的訪問和修改,想要控制那麼有幾種操作,今天我們就來講講第一種方法:線程同步塊或者線程同步方法(synchronized)

實例

  1. 下麵舉一個例子說明synchronized關鍵字的使用

線程同步方法

public class Sychor {
    public void insert(Thread thread) {
        for (int i = 0; i < 10; i++) {
            System.out.println(thread.getName() + "輸出:  " + i);
        }

    }

    public static void main(String[] args) {
        final Sychor sychor = new Sychor();

        Thread t1 = new Thread() {
            public void run() {
                sychor.insert(Thread.currentThread());
            };
        };

        Thread t2 = new Thread() {
            public void run() {
                sychor.insert(Thread.currentThread());
            };
        };

        t1.start();
        t2.start();
    }
}

其中輸出結果為下圖

運行結果

從上面的結果可以看出這裡的兩個線程是同時執行insert()方法的,下麵我們在原有的代碼上添加synchronized關鍵字看看效果如何,代碼如下:

public class Sychor {
    public synchronized void insert(Thread thread) {
        for (int i = 0; i < 10; i++) {
            System.out.println(thread.getName() + "輸出:  " + i);
        }

    }

    public static void main(String[] args) {
        final Sychor sychor = new Sychor();

        Thread t1 = new Thread() {
            public void run() {
                sychor.insert(Thread.currentThread());
            };
        };

        Thread t2 = new Thread() {
            public void run() {
                sychor.insert(Thread.currentThread());
            };
        };

        t1.start();
        t2.start();
    }
}

上面程式的運行結果我就不列出來,自己可以試試,總之就是加上了synchronized關鍵字使得線程是一個一個的執行的,只有先執行完一個線程才能執行了另外一個線程。

線程同步塊

當然上面的我們使用的是線程同步方法,我們可以使用線程同步塊,這兩個相比線程同步塊更加靈活,只需要將需要同步的代碼放在同步塊中即可,代碼如下;

public class Sychor {
    public void insert(Thread thread) {
        synchronized (this) {
            for (int i = 0; i < 10; i++) {
                System.out.println(thread.getName() + "輸出:  " + i);
            }
            
        }
        

    }

    public static void main(String[] args) {
        final Sychor sychor = new Sychor();

        Thread t1 = new Thread() {
            public void run() {
                sychor.insert(Thread.currentThread());
            };
        };

        Thread t2 = new Thread() {
            public void run() {
                sychor.insert(Thread.currentThread());
            };
        };

        t1.start();
        t2.start();
    }
}

從上面的代碼中可以看出這種方式更加靈活,只需要將需要同步的代碼方法在同步塊中,不需要同步的代碼放在外面

詳細原因

  1. 我們知道每一個對象都有一把鎖,當我們使用線程同步方法或者線程同步塊的時候實際上獲得是對象的唯一的一把鎖,當一個線程獲得了這唯一的鎖,那麼其他的線程只能拒之門外了,註意這裡我們說是一個對象,也就是說是同一個對象,如果是不同的對象,那麼就不起作用了,因為不同對象有不同的對象鎖,比如我們將上面的程式改成如下:

public class Sychor {
    public void insert(Thread thread) {
        synchronized (this) {
            for (int i = 0; i < 10; i++) {
                System.out.println(thread.getName() + "輸出:  " + i);
            }
        }

    }

    public static void main(String[] args) {

        //第一個線程
        Thread t1 = new Thread() {
            public void run() {
                Sychor sychor = new Sychor();   //在run() 方法中創建一個對象
                sychor.insert(Thread.currentThread());
            };
        };

        //第二個線程
        Thread t2 = new Thread() {
            public void run() {
                Sychor sychor = new Sychor();  //創建另外的一個對象
                sychor.insert(Thread.currentThread());
            };
        };

        t1.start();
        t2.start();
    }
}

從上面的結果可知,此時線程同步塊根本不起作用,因為他們調用的是不同對象的insert方法,獲得鎖是不一樣的

  1. 上面我們已經說過一個對象有一把鎖,線程同步方法和線程同步塊實際獲得的是對象的鎖,因此線程同步塊的括弧中填入的是this,我們都知道this在一個類中的含義
  1. 一個類也有唯一的一把鎖,我們前面說的是使用對象調用成員方法,現在如果我們要調用類中的靜態方法,那麼我們可以使用線程同步方法或者同步塊獲得類中的唯一一把鎖,那麼對於多個線程同時調用同一個類中的靜態方法就可以實現控制了,代碼如下:
public class Sychor {
    // 靜態方法
    public static synchronized void insert(Thread thread)  
    {
        for(int i=0;i<10;i++)
        {
            System.out.println(thread.getName()+"輸出     "+i);
        }
    }
    public static void main(String[] args) {

        //第一個線程
        Thread t1 = new Thread() {
            public void run() {
                Sychor.insert(Thread.currentThread());  //直接使用類調用靜態方法
            };
        };

        //第二個線程
        Thread t2 = new Thread() {
            public void run() {
                Sychor.insert(Thread.currentThread());   //直接使用類調用靜態方法
            };
        };

        t1.start();
        t2.start();
    }
}

註意

  1. 要想實現線程安全和同步控制,如果執行的是非static同步方法或者其中的同步塊,那麼一定要使用同一個對象,如果調用的是static同步方法或者其中的同步塊那麼一定要使用同一個類去調用
  1. 如果一個線程訪問的是static同步方法,而另外一個線程訪問的是非static的同步方法,此時這兩個是不會發生衝突的,因為一個是類的鎖,一個是對象的鎖
  1. 如果使用線程同步塊,那麼同步塊中的代碼是控制訪問的,但是外面的代碼是所有線程都可以訪問的
  1. 當一個正在執行同步代碼塊的線程出現了異常,那麼jvm會自動釋放當前線程所占用的鎖,因此不會出現由於異常導致死鎖的現象

參考文章

http://www.cnblogs.com/dolphin0520/p/3923737.html


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 泛型是.net 2.0就有的特性,泛型在我們的平常的開發過程中使用得也非常多,為了更深刻地理解泛型,這篇文章就來總結一下。 什麼是泛型 可以用下麵2點來概括: 1,首先泛型也是一種類型(可以通過IL代碼看出來)。 2,泛型提供了類型參數化的能力,允許用不同的類型進行實例化,可以簡單地理解為:泛型是類 ...
  • 之前,有發過不少PC端快速開發的文章,但是現在越來越多的項目需要PC和移動端結合,所以,我給大家介紹一下之前快速開發平臺的APP版本,希望大家能夠體驗一下,發表一下自己的看法。下載的話,在官網:www.learun.cn上有APP二維碼,掃描即可下載體驗。 獻上《線上體驗Demo地址》希望大家也能從 ...
  • 最近在一個項目中涉及到了虛擬目錄與UNC路徑的問題,總結出來分享給大家。 問題描述 某客戶定製化項目(官網),有一個圖片上傳的功能。客戶的Web機器有10台,通過F5負載均衡分攤請求。 假設這10台機器的代號分別為:#1,#2,#3,#4,#5,#6,#7,#8,#9,#10 在沒有應用虛擬目錄時, ...
  • C++與matlab混合編程——C++調用MATLAB函數 筆者最近在從事一個MFC相關的項目,要求將用Matlab實現的演算法通過應用MFC製作成一個小應用。其中有一部分內容需要求一個多元函數的最值。通過網路,我找到了兩個C++優化庫,dlib與MIDACO_Project ,可是這兩個庫中的優化函 ...
  • 安裝VS2013時,如何避開IE10的限制 VS就會告訴我們目前環境不適合安裝VS2013,必須升級IE版本到IE10. 如果不想安裝IE10,有沒有辦法呢? 答案肯定是有的。 將下麵一段文字,儲存為1.bat文檔,然後以管理員身份執行. 執行後,視窗回自動退出,然後你重新安裝VS即可。 ...
  • expect腳本內容 [root@ward-test script]# cat scp.exp #!/usr/bin/expect#Writen by : Ward/[email protected]#ScriptName: scp.expset timeout 10 set host [lindex ...
  • Windows下用Composer引入官方GitHub擴展包 1. 當你打開威武RC4版本的鏈接的時候,往下拉你可以看到這個,然後你要做的就是想到,百度Composer,看看是個什麼鬼,別想太多,跟著我走。接下來點擊Composer中文文檔,再點擊下載你會看到下載完後,點開如圖所示點擊next後發現 ...
  • Akka是一種消息驅動運算模式,它實現跨JVM程式運算的方式是通過能跨JVM的消息系統來調動分佈在不同JVM上ActorSystem中的Actor進行運算,前題是Akka的地址系統可以支持跨JVM定位。Akka的消息系統最高境界可以實現所謂的Actor位置透明化,這樣在Akka編程中就無須關註Act ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...