Java中線程總結

来源:http://www.cnblogs.com/xie-zhonglai/archive/2017/05/26/java_mul_thread.html
-Advertisement-
Play Games

本文簡要介紹在 Java 世界中, 線程相關知識。主要包含 線程的創建與銷毀;線程安全與同步;線程通訊;註意本文沒有什麼高深新知識,只緣起前段時間在翻看項目代碼的時候,發現有些同學對此有諸多誤解,故在此稍微整理一下,以幫助類似同學,同時警醒一下自己。 ...


本文簡要介紹在 Java 世界中, 線程相關知識。主要包含 線程的創建與銷毀;線程安全與同步;線程通訊;註意本文沒有什麼高深新知識,只緣起前段時間在翻看項目代碼的時候,發現有些同學對此有諸多誤解,故在此稍微整理一下,以幫助類似同學,同時警醒一下自己。

1.  線程的創建和銷毀;

a) .創建線程可以通過繼承 Thread 類 或 實現 Runnable 介面, 並重寫 run() 方法, 其中的run() 方法即是本線程需要執行的內容.

b). 相比於單獨繼承 Thread ,Runnable介面配合 Thread 實現會更靈活,並可以通過共用一個Runnable介面實例,在Thread中共用資源.

c). 至於線程銷毀,不推薦使用 Thread.Stop()方法, 此方法在使用不當情況下會出現死鎖,更多的時候推薦在run()方法中使用額外變數(或條件)結束此方法即可.

 2. 線程安全與同步;

a). 對於需要遵循ACID原子一致性的代碼段, 可以通過 synchronized(lockKey){} 代碼塊鎖定;

b). 同時 synchronized 關鍵字可以用來修飾一個方法,表示整個方法都需要遵循ACID原子一致性,值得註意的是,此時其實的lockKey等效於this關鍵字;

b). 在鎖定的代碼塊中推薦再進行一次必要的條件判斷。

  3. 線程通訊,在java的世界中可以藉助 wait() notify() notifyAll() 這三個方法來完成,這三個方法定義在Object類中,因此所有的對象都可以使用.

  4.下麵通過簡單的幾個代碼片段來加以說明

  a). 演示線程創建與銷毀,及線程安全與同步

 

 1 public class ThreadTest implements Runnable {
 2     private boolean stop;                   //是否需要停止運行
 3     private int tiketCount = 100000;        //總票數
 4     private boolean lockTypeIsMethod = true; //是否是提供方法鎖定還是代碼塊
 5 
 6     public boolean isStop() {
 7         return stop;
 8     }
 9     public void setStop(boolean stop) {
10         this.stop = stop;
11     }
12 
13     public boolean isLockTypeMethod() {
14         return lockTypeIsMethod;
15     }
16     public void setLockTypeIsMethod(boolean lockTypeIsMethod) {
17         this.lockTypeIsMethod = lockTypeIsMethod;
18     }
19 
20     @Override
21     public void run() {
22         while (tiketCount > 0 && !stop) {
23             try {
24                 Thread.sleep(50);    //延時,方便演示
25             } catch (InterruptedException e) {
26                 e.printStackTrace();
27             }
28 
29             //如果是通過鎖定方法
30             if (lockTypeIsMethod) {
31                 sale();
32             } else {
33                 synchronized (this) {
34                     if (tiketCount > 0 && !stop) {
35                         System.out.println("使用代碼塊鎖定:threadId="
36                                 + Thread.currentThread().getName() + ",ticketNO:" + tiketCount--);
37                     }
38                 }
39             }
40         }
41     }
42 
43     public synchronized void sale() {
44         if (tiketCount > 0 && !stop) {
45             System.out.println("使用方法鎖定:threadId="
46                     + Thread.currentThread().getName() + ",ticketNO:" + tiketCount--);
47         }
48     }
49 }
線程定義類

 

 1 public static void main(String[] args) throws InterruptedException {
 2     ThreadTest threadTest = new ThreadTest();   //共用變數ThreadTest
 3     //啟用四個線程
 4     new Thread(threadTest).start();
 5     new Thread(threadTest).start();
 6     new Thread(threadTest).start();
 7     new Thread(threadTest).start();
 8     //模擬設置共用變數,
 9     // 1.交替使用方法體和代碼塊來進行線程同步實驗
10     // 2.模擬線程停止
11     for (int i = 0; i < 100; i++) {
12         Thread.sleep(1000);
13         threadTest.setLockTypeIsMethod(i % 2 == 0);
14         if (i == 50) {
15             threadTest.setStop(true);
16         }
17     }
18 }
調用端

  b). 演示線程通訊,本處模擬兩個線程以生產和消費者角色讀寫一個集合的示例,其中當集合中有數據的時候通知消費者處理數據,處理完後通知生產者往集合中放入數據

 1 //數據倉庫
 2 public class DataRepository {
 3     private List<String> data = new ArrayList<>();
 4     private boolean hasData;
 5 
 6     public boolean HasData() {
 7         return hasData;
 8     }
 9 
10     public void setHasData(boolean hasData) {
11         this.hasData = hasData;
12     }
13 
14     //放入數據
15     public synchronized void put(List<String> data) throws InterruptedException {
16         //生產者放入數據的時候,如果還有數據則等待.
17         if (hasData) {
18             wait();
19         }
20         this.data = data;
21         hasData = true;
22         //放入完畢後通知消費者
23         notify();
24     }
25 
26     //讀取數據
27     public synchronized List<String> get() throws InterruptedException {
28         //沒有數據則等待
29         if (!hasData) {
30             wait();
31         }
32         //獲取數據副本返回
33         List<String> rs = new ArrayList<>(data);
34         data.clear();
35         hasData = false;
36         notify();
37         return rs;
38     }
39 }
數據倉庫
 1 public class Producer implements  Runnable {
 2     private DataRepository dataRepository;      //數據倉庫
 3     public Producer(DataRepository dataRepository) {
 4         this.dataRepository = dataRepository;
 5     }
 6 
 7     public void run() {
 8         while (true) {
 9             try {
10                 Thread.sleep(1000);
11             } catch (InterruptedException e) {
12                 e.printStackTrace();
13             }
14             List<String> temp = new ArrayList<>();
15             temp.add("------------");
16             temp.add("第一個數據");
17             temp.add("第二個數據");
18             temp.add("第三個數據");
19             temp.add("第四個數據");
20             temp.add("------------");
21             try {
22                 dataRepository.put(temp);
23             } catch (InterruptedException e) {
24                 e.printStackTrace();
25             }
26         }
27     }
28 }
生產者
 1 public class Consumer implements Runnable {
 2     private DataRepository dataRepository;
 3 
 4     public Consumer(DataRepository dataRepository) {
 5         this.dataRepository = dataRepository;
 6     }
 7 
 8     public void run() {
 9         while (true) {
10             try {
11                 Thread.sleep(1000);
12             } catch (InterruptedException e) {
13                 e.printStackTrace();
14             }
15             try {
16                 List<String> data=dataRepository.get();
17                 if(data!=null&&!data.isEmpty()){
18                     for (String temp :data){
19                         System.out.println(temp);
20                     }
21                 }
22             } catch (InterruptedException e) {
23                 e.printStackTrace();
24             }
25         }
26     }
27 }
消費者
1 public class Client {
2     public static void main(String[] args) throws InterruptedException {
3    
4         DataRepository dr=new DataRepository();
5         new Thread(new Producer(dr)).start();   //啟動生產者線程
6         new Thread(new Consumer(dr)).start();   //啟動消費者線程
7     }
8 }
調用端

 

後記:

  a). 多線程屬於較基礎的知識,我們首先需要瞭解其最基本的概念,才能在項目中游刃有餘的應用;

  b).不管是什麼語言,其所需要的理論支持均大同小異;

  c).回到最初的那個概念,在多線程中,能不需要線程互相通訊就儘量不要用,能不同步就儘量不要使用線程同步,能不使用多線程就儘量不要使用多線程,說得有些含糊,各位自己去參悟吧.

 


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

-Advertisement-
Play Games
更多相關文章
  • 一、新建一個html頁面,如註冊頁面"Register.htm" 二、新建一js文件,如:reg.js 三、處理ajax請求 方法一:手動拼接json字元串 新建一般處理程式,如:Register.ashx 方法二:使用Json.NET工具來將C#對象轉換json輸出 1、新建信息類“Msg.cs” ...
  • 前言 很多時候其實我們並不需要asp.net core自帶的那麼複雜的用戶系統,基於角色,各種概念,還得用EF Core,而且在web應用中都是把信息存儲到cookie中進行通訊(我不喜歡放cookie中,因為有次我在mac系統中的safari瀏覽器運行web應用時,碰到跨域cookie設不上,非要 ...
  • 回到目錄 TransactionScope是.net環境下的事務,可以提升為分散式事務,這些知識早在很久前就已經說過了,今天不再說它,今天主要談談Savechanges()這個方法在TransactionScope塊里的作用,我們知識TransactionScope只有顯示的提交動作而沒有回滾,那麼 ...
  • 首先並不是每個事件的e參數都有上述兩個屬性。 e.Cancel:獲取或設置指示是否應取消事件的值;e.Handled:獲取或設置一個值,該值指示是否處理過此事件。 下麵說說比較常見的場景: 1)e.cancel: ①視窗關閉,比如用戶點擊視窗右上角想關閉,但代碼里彈出確認框讓用戶確認是否真的想退出, ...
  • 1、Messager交互結構和消息類型 銜接上篇,Messeger是信使的意思,顧名思義,他的目是用於View和ViewModel 以及 ViewModel和ViewModel 之間的消息通知和接收。 Messenger類用於應用程式的通信,接受者只能接受註冊的消息類型,另外目標類型可以被指定,用S ...
  • 某天 , 在review項目中代碼的時候, 發現有哥們直接通過 Database.ExecuteSqlCommand("select * from order_info where company like '%abc%' ")的方式與資料庫查詢, 私下問其是否知道這個方法還有一個帶有params ...
  • 一、cron表達式簡單介紹和下載 1、在上一篇博客“Quartz.net 定時任務之簡單任務”中,我簡單介紹了quartz的使用,而這篇博客我將介紹cron的具體使用(不足之處望大神斧正) 1、cron是為了方便編寫定時執行作業時間擴展出來的插件,這個有很多版本(網頁版,窗體版等),cron表達式的 ...
  • 首先看一下無參的構造方法: 在未指定容量大小時,會將final的DEFAULTCAPACITY_EMPTY_ELEMENTDATA給elementData,這樣的好處是無論多少次實例化無參ArrayList初始的保存對象都是固定的,而不必每次都創建一個新的Object數組. 這樣需要在每次做添加操作 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...