Java多線程(三)

来源:https://www.cnblogs.com/xiaozhao01/archive/2022/08/05/16555777.html
-Advertisement-
Play Games

Java多線程(三) 五、線程的通信 5.1 wait() 與 notify() 和 notifyAll() 介紹: wait():令當前線程掛起並放棄CPU、同步資源並等待,使別的線程可訪問並修改共用資源,而當前線程排隊等候其他線程調用notify() 或 notifyAll() 方法喚醒,喚醒後 ...


Java多線程(三)

目錄

五、線程的通信

5.1 wait() 與 notify() 和 notifyAll() 介紹:

  1. wait():令當前線程掛起並放棄CPU、同步資源並等待,使別的線程可訪問並修改共用資源,而當前線程排隊等候其他線程調用notify() 或 notifyAll() 方法喚醒,喚醒後等待重新獲得對監視器的所有權後才能繼續執行。
  2. notify():喚醒正在排隊等待同步資源的線程中優先順序最高者結束等待。
  3. notifyAll ():喚醒正在排隊等待資源的所有線程結束等待。
  • 這三個方法只有在 synchronized 方法 或 synchronized 代碼塊中才能使用,否則會報 java.lang.IllegalMonitorStateException異常。

  • 因為這三個方法必須有鎖對象調用,而任意對象都可以作為 synchronized 的同步鎖, 因此這三個方法只能在Object類中聲明。

5.2 wait() 的使用:

  1. 在當前線程中調用方法: 對象名.wait() 。
  2. 使當前線程進入等待(某對象)狀態 ,直到另一線程對該對象發出 notify (或notifyAll) 為止。
  3. 調用方法的必要條件:當前線程必須具有對該對象的監控權(加鎖)。
  4. 調用此方法後,當前線程將釋放對象監控權 ,然後進入等待。
  5. 在當前線程被 notify 後,要重新獲得監控權,然後從斷點處繼續代碼的執行。

5.3 notify() / notifyAll() 的使用

  1. 在當前線程中調用方法: 對象名.notify()。
  2. 功能:喚醒等待該對象監控權的一個/所有線程。
  3. 調用方法的必要條件:當前線程必須具有對該對象的監控權(加鎖)。

5.4 經典例題:生產者/消費者問題

題目描述:

生產者(Productor)將產品交給店員(Clerk),而消費者(Customer)從店員處取走產品,店員一次只能持有固定數量的產品(比如:20),如果生產者試圖生產更多的產品,店員會叫生產者停一下,如果店中有空位放產品了再通知生產者繼續生產;如果店中沒有產品了,店員會告訴消費者等一下,如 果店中有產品了再通知消費者來取走產品。

class Clerk{

    private int productCount = 0;
    //生產產品
    public synchronized void produceProduct() {

        if(productCount < 20){
            productCount++;
            System.out.println(Thread.currentThread().getName() + ":開始生產第" + productCount + "個產品");
            
            notify();

        }else{
            //等待
            try {
                wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
    //消費產品
    public synchronized void consumeProduct() {
        if(productCount > 0){
            System.out.println(Thread.currentThread().getName() + ":開始消費第" + productCount + "個產品");
            productCount--;

            notify();
        }else{
            //等待
            try {
                wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

class Producer extends Thread{//生產者

    private Clerk clerk;

    public Producer(Clerk clerk) {
        this.clerk = clerk;
    }

    @Override
    public void run() {
        System.out.println(getName() + ":開始生產產品.....");

        while(true){
            
            try {
                Thread.sleep(10);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }

            clerk.produceProduct();
        }
    }
}

class Consumer extends Thread{//消費者
    private Clerk clerk;

    public Consumer(Clerk clerk) {
        this.clerk = clerk;
    }

    @Override
    public void run() {
        System.out.println(getName() + ":開始消費產品.....");

        while(true){

            try {
                Thread.sleep(20);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }

            clerk.consumeProduct();
        }
    }
}

public class ProductTest {

    public static void main(String[] args) {
        Clerk clerk = new Clerk();

        Producer p1 = new Producer(clerk);
        p1.setName("生產者1");

        Consumer c1 = new Consumer(clerk);
        c1.setName("消費者1");
        Consumer c2 = new Consumer(clerk);
        c2.setName("消費者2");

        p1.start();
        c1.start();
        c2.start();
    }
}

六、JDK5.0新增多線程創建方式

6.1 多線程的創建方式之三:實現Callable介面

  1. 創建一個實現 Callable 的實現類
  2. 實現call方法,將此線程需要執行的操作聲明在call() 中。
  3. 創建 Callable 介面實現類的對象。
  4. 將此 Callable 介面實現類的對象作為傳遞到 FutureTask 構造器中,創建 FutureTask 的對象。
  5. 將 FutureTask 的對象作為參數傳遞到 Thread 類的構造器中,創建 Thread 對象,並調用 start() 。
  6. 獲取 Callable 中 call 方法的返回值。(可選)
//1.創建一個實現Callable的實現類
class NumThread implements Callable{
    //2.實現call方法,將此線程需要執行的操作聲明在call()中
    @Override
    public Object call() throws Exception {
        int sum = 0;
        for (int i = 1; i <= 100; i++) {
            if(i % 2 == 0){
                System.out.println(i);
                sum += i;
            }
        }
        return sum;
    }
}

public class ThreadNew {
    public static void main(String[] args) {
        
        //3.創建Callable介面實現類的對象
        NumThread numThread = new NumThread();
        //4.將此Callable介面實現類的對象作為傳遞到FutureTask構造器中,創建FutureTask的對象
        FutureTask futureTask = new FutureTask(numThread);
        //5.將FutureTask的對象作為參數傳遞到Thread類的構造器中,創建Thread對象,並調用start()
        new Thread(futureTask).start();

        try {
            //6.獲取Callable中call方法的返回值
            //get()返回值即為FutureTask構造器參數Callable實現類重寫的call()的返回值。
            Object sum = futureTask.get();
            System.out.println("總和為:" + sum);
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (ExecutionException e) {
            e.printStackTrace();
        }
    }
}
  • 與使用 Runnable 相比, Callable功能更強大些:

    1. 相比run()方法,可以有返回值。
    2. 方法可以拋出異常。
    3. 支持泛型的返回值。
    4. 需要藉助 FutureTask 類,比如獲取返回結果。
  • Future 介面:

    1. 可以對具體Runnable、Callable任務的執行結果進行取消、查詢是否完成、獲取結果等。
    2. FutrueTask 是 Futrue 介面的唯一的實現類。
    3. FutureTask 同時實現了Runnable, Future介面。它既可以作為 Runnable 被線程執行,又可以作為 Future 得到 Callable 的返回值。

6.2 多線程的創建方式之四:使用線程池

  1. 線程池的引入:

    經常創建和銷毀、使用量特別大的資源,比如併發情況下的線程, 對性能影響很大。

  2. 線程池的思路:

    提前創建好多個線程,放入線程池中,使用時直接獲取,使用完放回池中。可以避免頻繁創建銷毀、實現重覆利用。

  3. 使用線程池的好處:

    • 提高響應速度(減少了創建新線程的時間)

    • 降低資源消耗(重覆利用線程池中線程,不需要每次都創建)

    • 便於線程管理

      corePoolSize:核心池的大小

      maximumPoolSize:最大線程數

      keepAliveTime:線程沒有任務時最多保持多長時間後會終止

  4. 線程池的使用:

(1)提供指定線程數量的線程池。

(2)執行指定的線程的操作,需要提供實現 Runnable 介面或 Callable 介面實現類的對象。

(3)關閉連接池。

class NumberThread implements Runnable{

    @Override
    public void run() {
        for(int i = 0;i <= 100;i++){
            if(i % 2 == 0){
                System.out.println(Thread.currentThread().getName() + ": " + i);
            }
        }
    }
}

class NumberThread1 implements Runnable{

    @Override
    public void run() {
        for(int i = 0;i <= 100;i++){
            if(i % 2 != 0){
                System.out.println(Thread.currentThread().getName() + ": " + i);
            }
        }
    }
}

public class ThreadPool {

    public static void main(String[] args) {
        //1. 提供指定線程數量的線程池
        ExecutorService service = Executors.newFixedThreadPool(10);
      
        //2.執行指定的線程的操作。需要提供實現Runnable介面或Callable介面實現類的對象
        service.execute(new NumberThread());//適合適用於Runnable
        service.execute(new NumberThread1());//適合適用於Runnable
//        service.submit(Callable callable);//適合使用於Callable
        
        //3.關閉連接池
        service.shutdown();
    }
}
  1. 線程池相關API
  • JDK 5.0 起提供了線程池相關API:ExecutorService 和 Executors 。
  • ExecutorService:真正的線程池介面。常見子類 ThreadPoolExecutor
//	執行任務/命令,沒有返回值,一般用來執行 Runnable
	void execute(Runnable command)
        
//	執行任務,有返回值,一般又來執行 Callable
    Future submit(Callable task)

//	關閉連接池
    void shutdown()
  • Executors:工具類、線程池的工廠類,用於創建並返回不同類型的線程池。
//	創建一個可根據需要創建新線程的線程池
	Executors.newCachedThreadPool()
        
//	創建一個可重用固定線程數的線程池        
	Executors.newFixedThreadPool(n)
        
//	創建一個只有一個線程的線程池          
    Executors.newSingleThreadExecutor() 
        
//	創建一個線程池,它可安排在給定延遲後運行命令或者定期地執行。 
	Executors.newScheduledThreadPool(n)
  1. 線程池的屬性設置

    • ExecutorService 是一個介面,裡面並沒有設置線程池屬性的方法,故不能用該介面的對象來對線程池進行屬性設置。

    • 設置線程池屬性需要通過創建 ExecutorService 介面的實現類 ThreadPoolExecutor 的對象,調用該實現類的方法進行設置。

        ThreadPoolExecutor service1 = (ThreadPoolExecutor) service;
        //設置線程池的屬性
        service1.setCorePoolSize(15);
        service1.setKeepAliveTime();

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

-Advertisement-
Play Games
更多相關文章
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 在javascript中使用Three.js設計並且實現3D場景是一個很有意思的事情,因為在瀏覽器中就能夠渲染出3D場景,非常簡單和輕便。接下來就總結以下我學習Three.js過程中的心得。 1 建立基本場景 在Three.js中有三要素 ...
  • 安裝 首先需要安裝好NPM,然後使用下列的命令一次進行 npm install -g @vue/cli //安裝腳手架 vue create xx //創建Vue項目,xx為要創建的項目名 創建項目後,使用 cd 進入目錄 npm run serve //啟動項目 目錄 Pubulic:靜態文件 s ...
  • 1. 事件起因 之前做一個駕照考題的項目,有一個這樣的問題,每當我選好了科目和駕照類型後(如圖1),點擊開始考試就會跳到考試頁面(Test.tsx),並且在Test組件中對我架設的中間層發起請求獲取數據(如圖2)。 如果用戶手滑的話不小心點到了左上角的返回,或者狠狠地把屏幕往右滑動一下的話,都會返回 ...
  • 1. 事件起因 事情是這樣的,我之前在做一個仿網易雲的項目,想實現一個功能,就是點擊歌曲列表組件(MusicList)中每一個item時,底部播放欄(FooterMusic)就能夠獲取我所點擊的這首歌曲的所有信息(如圖1到圖2),但是底部播放欄是直接放在外殼組件App.vue中固定定位的,歌曲列表組 ...
  • 引言 當今社會對軟體需求在相當長的時間里將保持旺盛,而軟體開發周期長、個性化難、順應需求變更不變,如何可以才能將軟體開發定製變得簡單方便快捷呢? 1. 軟體開發設計現狀 目前軟體的開發設計都是定向開發,即根據項目需求將相關的數據關係、業務邏輯、功能模塊及介面插件等揉合在一起並與人機交互整體開發(若涉 ...
  • 單元測試JUnit 單元測試的目的是針對方法進行測試, **JUnit的兩個要點:**①必須是公開的,無參數,無返回值的方法 ②測試方法必須使用@Test註解標記 public class JUnitTest { @Test public void Testusername() { way way ...
  • 目錄 一.簡介 二.效果演示 三.源碼下載 四.猜你喜歡 零基礎 OpenGL (ES) 學習路線推薦 : OpenGL (ES) 學習目錄 >> OpenGL ES 基礎 零基礎 OpenGL (ES) 學習路線推薦 : OpenGL (ES) 學習目錄 >> OpenGL ES 轉場 零基礎 O ...
  • 多線程併發 在多核CPU中,利用多線程併發編程,可以更加充分地利用每個核的資源 在Java中,一個應用程式對應著一個JVM實例(也有地方稱為JVM進程),如果程式沒有主動創建線程,則只會創建一個主線程。但這不代表JVM中只有一個線程,JVM實例在創建的時候,同時會創建很多其他的線程(比如垃圾收集器線 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...