Java多線程學習(Day02)

来源:https://www.cnblogs.com/CooperCode/archive/2023/09/17/17709026.html
-Advertisement-
Play Games

目錄 線程簡介 線程實現(重點) 線程狀態 線程同步(重點) 線程通信問題 線程實現: 方式一:繼承Thread類 /** * TODO * @author 清蓮孤舟 * @CreateDate 2023/9/17/9:28 * 創建線程的方式一:通過繼承Thread類實現 */ //繼承Threa ...


目錄

線程實現:

方式一:繼承Thread類

/**
 * TODO
 * @author 清蓮孤舟
 * @CreateDate 2023/9/17/9:28
 * 創建線程的方式一:通過繼承Thread類實現
 */
//繼承Thread類
public class demo01 extends Thread {
//重寫run方法
    @Override
    public void run() {
        for (int i = 0; i < 20; i++) {
            System.out.println("副線程"+i);
        }
    }
//main主線程
    public static void main(String[] args) {
        demo01 demo01 = new demo01();
        //start方法啟用線程
        demo01.start();
        //run()方法啟用線程
        demo01.run();
        for (int i = 0; i < 20; i++) {
            System.out.println("主線程"+i);
        }
    }
}

  我們先調用run方法運行(註意run()方法在主函數中的順序):

副線程0    副線程1    副線程2    副線程3    副線程4    副線程5    副線程6    副線程7    副線程8    副線程9    
副線程10    副線程11    副線程12    副線程13    副線程14    副線程15    副線程16    副線程17    副線程18    副線程19

主線程0    主線程1    主線程2    主線程3    主線程4    主線程5    主線程6    主線程7    主線程8    主線程9    
主線程10    主線程11    主線程12    主線程13    主線程14    主線程15    主線程16    主線程17    主線程18    主線程19    

  可以看出程式先運行完run線程再運行main主線程。符合程式按序執行的規制。同理如果將run()方法放在主函數的最後,那麼先輸出的將是"主線程"隨後是"副線程"。

  接下來我們調用start()方法:

主線程0    副線程0    主線程1    主線程2    主線程3    主線程4    副線程1    主線程5    主線程6    主線程7    副線程2    主線程8    主線程9    主線程10   主線程11   主線程12   副線程3    
主線程13 主線程14 主線程15 主線程16 主線程17 主線程18 主線程19 副線程4 副線程5 副線程6 副線程7 副線程8 副線程9 副線程10 副線程11 副線程12 副線程13
副線程14 副線程15 副線程16 副線程17 副線程18 副線程19

  顯而易見程式並不是按序執行的,而是交替並行的。這便是上節中介紹的併發,在一段時間內交替完成。

  (PS:如果你在運行時發現輸出結果並不是交替的,那麼請將for迴圈次數增加,20次迴圈對於你的電腦來說太簡單了)

方式二:實現Runnable介面(重點)

/**
 * TODO
 * @author Cooper
 * @CreateDate 2023/9/17/10:26
 * 創建線程方式二:實現Runnable介面
 */
public class demo02 implements Runnable{
//   同樣的重寫run()方法
    @Override
    public void run() {
        for (int i = 0; i < 20; i++) {
            System.out.print("副線程"+i+"\t");
        }
    }
    public static void main(String[] args) {
        demo02 demo02 = new demo02();
//        創建線程對象,通過線程對象開啟線程
        Thread thread = new Thread(demo02);
        thread.start();
//      可以簡寫為該寫法new Thread(demo02).start();
        for (int i = 0; i < 20; i++) {
            System.out.print("主線程"+i+"\t");
        }
    }
}

  源碼解析:

//Runnable源碼 只有一個run()方法
public interface Runnable {
    public abstract void run();
}
//Thread方法 需要傳入一個實現Runnable的對象
public Thread(Runnable target) {
    init(null, target, "Thread-" + nextThreadNum(), 0);
}

  運行結果:

主線程0    副線程0    主線程1    副線程1    主線程2    主線程3    主線程4    主線程5    副線程2    主線程6    
主線程7    主線程8    副線程3    主線程9    副線程4    主線程10    主線程11    主線程12    主線程13    主線程14    
主線程15    主線程16    主線程17    副線程5    主線程18    副線程6    副線程7    主線程19    副線程8    副線程9    
副線程10    副線程11    副線程12    副線程13    副線程14    副線程15    副線程16    副線程17    副線程18    副線程19    

線程併發案例:

  這裡使用狂神的案例:

 1 /**
 2  * TODO
 3  *
 4  * @author Cooper
 5  * @CreateDate 2023/9/17/10:42
 6  */
 7 public class demo03 implements Runnable{
 8     private int TicketNumber = 10;
 9     @Override
10     public void run() {
11         while (true){
12             if(TicketNumber==0){
13                 break;
14             }
15             try {
16                 //線程暫停200ms
17                 Thread.sleep(200);
18             } catch (InterruptedException e) {
19                 e.printStackTrace();
20             }
21             //Thread.currentThread().getName()獲取當前線程的名字
22             System.out.println(Thread.currentThread().getName()+"===>拿到了第"+TicketNumber--+"票");
23         }
24     }
25     public static void main(String[] args) {
26         demo03 ticket = new demo03();
27         new Thread(ticket,"小張").start();
28         new Thread(ticket,"小李").start();
29         new Thread(ticket,"小王").start();
30     }
31 }

  運行結果:

小李===>拿到了第8票
小張===>拿到了第10票
小王===>拿到了第9票
小王===>拿到了第6票
小李===>拿到了第7票
小張===>拿到了第7票
小李===>拿到了第3票
小王===>拿到了第4票
小張===>拿到了第5票
小李===>拿到了第2票
小王===>拿到了第2票
小張===>拿到了第2票
小張===>拿到了第1票
小王===>拿到了第1票
小李===>拿到了第1票

  此時我們發現了一些問題--->同一張票被兩個人拿到了! 這就是線程不安全,數據紊亂。後序我們將通過線程同步解決這個問題。

方式三:實現callable介面(瞭解即可)

 1 import java.util.ArrayList;
 2 import java.util.List;
 3 import java.util.concurrent.*;
 4 /**
 5  * TODO
 6  *
 7  * @author Cooper
 8  * @CreateDate 2023/9/17/14:52
 9  */
10 public class demo04 implements Callable<List<String>> {
11     @Override
12     public List<String> call() throws Exception {
13         List<String> result = new ArrayList<>();
14         for (int i = 0; i < 5; i++) {
15             result.add(Thread.currentThread().getName()+"===>"+i);
16             System.out.print(Thread.currentThread().getName()+"===>"+i+"\t");
17         }
18         return result;
19     }
20     public static void main(String[] args) throws ExecutionException, InterruptedException {
21         demo04 test1 = new demo04();
22         demo04 test2 = new demo04();
23         demo04 test3 = new demo04();
24 //        創建服務開啟三個線程
25         ExecutorService service = Executors.newFixedThreadPool(3);
26         System.out.println("運行中:");
27 //        提交執行
28         Future<List<String>> r1 = service.submit(test1);
29         Future<List<String>> r2 = service.submit(test2);
30         Future<List<String>> r3 = service.submit(test3);
31 //        獲取結果
32         List<String> result1 = r1.get();
33         List<String> result2 = r2.get();
34         List<String> result3 = r3.get();
35         System.out.println();
36 
37         System.out.println("運行結束:");
38         System.out.println(result1);
39         System.out.println(result2);
40         System.out.println(result3);
41 //        關閉服務
42         service.shutdown();
43     }
44 }

  運行結果:

運行中:
pool-1-thread-2===>0    pool-1-thread-1===>0    pool-1-thread-2===>1    pool-1-thread-3===>0    
pool-1-thread-2===>2    pool-1-thread-1===>1    pool-1-thread-2===>3    pool-1-thread-1===>2    
pool-1-thread-3===>1    pool-1-thread-1===>3    pool-1-thread-2===>4    pool-1-thread-1===>4    
pool-1-thread-3===>2    pool-1-thread-3===>3    pool-1-thread-3===>4    
運行結束:
[pool-1-thread-1===>0, pool-1-thread-1===>1, pool-1-thread-1===>2, pool-1-thread-1===>3, pool-1-thread-1===>4]
[pool-1-thread-2===>0, pool-1-thread-2===>1, pool-1-thread-2===>2, pool-1-thread-2===>3, pool-1-thread-2===>4]
[pool-1-thread-3===>0, pool-1-thread-3===>1, pool-1-thread-3===>2, pool-1-thread-3===>3, pool-1-thread-3===>4]

 

本文來自博客園,作者:Lollipop_pro,轉載請註明原文鏈接:https://www.cnblogs.com/CooperCode/p/17709026.html


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

-Advertisement-
Play Games
更多相關文章
  • 在MySQL中,高級查詢是指使用更複雜的查詢語句和操作符來檢索和操作資料庫中的數據。高級查詢可以幫助您更精確地找到所需的信息,並提高查詢的效率和靈活性。 以下是高級查詢的一些常見應用場景和意義: 連接多個表:使用JOIN操作符將多個表連接起來,以便在一次查詢中獲取相關聯的數據。這對於在多個表之間建立 ...
  • 背景: 隨著項目體量越來越大,用戶群體越來越多,用戶的聲音也越來越明顯;關於應用發版之後用戶無感知,導致用戶用的是仍然還是老版本功能,除非用戶手動刷新,否則體驗不到最新的功能;這樣的體驗非常不好,於是我們團隊針對該問題給出了相應的解決方案來處理;技術棧:vue3+ts+vite+ant-design ...
  • 相比用戶停留時間短、用完即走的 Web 頁面,桌面 QQ 用戶在一次登錄後,可能會掛機一周以上,這段期間,如果沒有嚴格控制好 QQ 記憶體占用,那麼結果可能是用戶交互響應變慢、甚至 Crash。在系統監控工具里,高記憶體占用也會被直觀地反映出來,帶來不好的口碑。MAC QQ 灰度期間,也聽到了一些用戶關... ...
  • 介紹 ESLint 是一個根據方案識別並報告 ECMAScript/JavaScript 代碼問題的工具,其目的是使代碼風格更加一致並避免錯誤。在很多地方它都與 JSLint 和 JSHint 類似,除了: ESLint 使用 Espree 對 JavaScript 進行解析。 ESLint 在代碼 ...
  • 這是一個講解DDD落地的文章系列,作者是《實現領域驅動設計》的譯者滕雲。本文章系列以一個真實的並已成功上線的軟體項目——碼如雲(https://www.mryqr.com)為例,系統性地講解DDD在落地實施過程中的各種典型實踐,以及在面臨實際業務場景時的諸多取捨。 本系列包含以下文章: DDD入門 ...
  • 系統設計之緩存五種策略 當我們在架構中引入緩存時,緩存和資料庫之間的同步就變得不可避免。 讓我們看看如何保持數據同步的五種常見策略。 1)閱讀策略: 緩存在一邊 通讀2)寫策略:寫周圍 回信 寫通緩存策略經常組合使用。例如,write-around 通常與 cache-aside 一起使用,以確保緩 ...
  • 前一篇水文中,老周演示了 QAbstractItemModel 抽象類的繼承方法。其實,在 Qt 的庫裡面,QAbstractItemModel 類也派生了兩個基類,能讓開發者繼承起來【稍稍】輕鬆一些。 這兩個類是 QAbstractListModel 和 QAbstractTableModel。 ...
  • 註釋可以用來解釋Python代碼。註釋可以用來使代碼更易讀。註釋可以用來在測試代碼時防止執行。 創建註釋 註釋以#開始,Python會忽略它們: 示例:獲取您自己的Python註釋 # 這是一個註釋 print("Hello, World!") 註釋可以放在一行的末尾,Python會忽略行的其餘部分 ...
一周排行
    -Advertisement-
    Play Games
  • 1、預覽地址:http://139.155.137.144:9012 2、qq群:801913255 一、前言 隨著網路的發展,企業對於信息系統數據的保密工作愈發重視,不同身份、角色對於數據的訪問許可權都應該大相徑庭。 列如 1、不同登錄人員對一個數據列表的可見度是不一樣的,如數據列、數據行、數據按鈕 ...
  • 前言 上一篇文章寫瞭如何使用RabbitMQ做個簡單的發送郵件項目,然後評論也是比較多,也是準備去學習一下如何確保RabbitMQ的消息可靠性,但是由於時間原因,先來說說設計模式中的簡單工廠模式吧! 在瞭解簡單工廠模式之前,我們要知道C#是一款面向對象的高級程式語言。它有3大特性,封裝、繼承、多態。 ...
  • Nodify學習 一:介紹與使用 - 可樂_加冰 - 博客園 (cnblogs.com) Nodify學習 二:添加節點 - 可樂_加冰 - 博客園 (cnblogs.com) 介紹 Nodify是一個WPF基於節點的編輯器控制項,其中包含一系列節點、連接和連接器組件,旨在簡化構建基於節點的工具的過程 ...
  • 創建一個webapi項目做測試使用。 創建新控制器,搭建一個基礎框架,包括獲取當天日期、wiki的請求地址等 創建一個Http請求幫助類以及方法,用於獲取指定URL的信息 使用http請求訪問指定url,先運行一下,看看返回的內容。內容如圖右邊所示,實際上是一個Json數據。我們主要解析 大事記 部 ...
  • 最近在不少自媒體上看到有關.NET與C#的資訊與評價,感覺大家對.NET與C#還是不太瞭解,尤其是對2016年6月發佈的跨平臺.NET Core 1.0,更是知之甚少。在考慮一番之後,還是決定寫點東西總結一下,也回顧一下.NET的發展歷史。 首先,你沒看錯,.NET是跨平臺的,可以在Windows、 ...
  • Nodify學習 一:介紹與使用 - 可樂_加冰 - 博客園 (cnblogs.com) Nodify學習 二:添加節點 - 可樂_加冰 - 博客園 (cnblogs.com) 添加節點(nodes) 通過上一篇我們已經創建好了編輯器實例現在我們為編輯器添加一個節點 添加model和viewmode ...
  • 前言 資料庫併發,數據審計和軟刪除一直是數據持久化方面的經典問題。早些時候,這些工作需要手寫複雜的SQL或者通過存儲過程和觸發器實現。手寫複雜SQL對軟體可維護性構成了相當大的挑戰,隨著SQL字數的變多,用到的嵌套和複雜語法增加,可讀性和可維護性的難度是幾何級暴漲。因此如何在實現功能的同時控制這些S ...
  • 類型檢查和轉換:當你需要檢查對象是否為特定類型,並且希望在同一時間內將其轉換為那個類型時,模式匹配提供了一種更簡潔的方式來完成這一任務,避免了使用傳統的as和is操作符後還需要進行額外的null檢查。 複雜條件邏輯:在處理複雜的條件邏輯時,特別是涉及到多個條件和類型的情況下,使用模式匹配可以使代碼更 ...
  • 在日常開發中,我們經常需要和文件打交道,特別是桌面開發,有時候就會需要載入大批量的文件,而且可能還會存在部分文件缺失的情況,那麼如何才能快速的判斷文件是否存在呢?如果處理不當的,且文件數量比較多的時候,可能會造成卡頓等情況,進而影響程式的使用體驗。今天就以一個簡單的小例子,簡述兩種不同的判斷文件是否... ...
  • 前言 資料庫併發,數據審計和軟刪除一直是數據持久化方面的經典問題。早些時候,這些工作需要手寫複雜的SQL或者通過存儲過程和觸發器實現。手寫複雜SQL對軟體可維護性構成了相當大的挑戰,隨著SQL字數的變多,用到的嵌套和複雜語法增加,可讀性和可維護性的難度是幾何級暴漲。因此如何在實現功能的同時控制這些S ...