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
  • Dapr Outbox 是1.12中的功能。 本文只介紹Dapr Outbox 執行流程,Dapr Outbox基本用法請閱讀官方文檔 。本文中appID=order-processor,topic=orders 本文前提知識:熟悉Dapr狀態管理、Dapr發佈訂閱和Outbox 模式。 Outbo ...
  • 引言 在前幾章我們深度講解了單元測試和集成測試的基礎知識,這一章我們來講解一下代碼覆蓋率,代碼覆蓋率是單元測試運行的度量值,覆蓋率通常以百分比表示,用於衡量代碼被測試覆蓋的程度,幫助開發人員評估測試用例的質量和代碼的健壯性。常見的覆蓋率包括語句覆蓋率(Line Coverage)、分支覆蓋率(Bra ...
  • 前言 本文介紹瞭如何使用S7.NET庫實現對西門子PLC DB塊數據的讀寫,記錄了使用電腦模擬,模擬PLC,自至完成測試的詳細流程,並重點介紹了在這個過程中的易錯點,供參考。 用到的軟體: 1.Windows環境下鏈路層網路訪問的行業標準工具(WinPcap_4_1_3.exe)下載鏈接:http ...
  • 從依賴倒置原則(Dependency Inversion Principle, DIP)到控制反轉(Inversion of Control, IoC)再到依賴註入(Dependency Injection, DI)的演進過程,我們可以理解為一種逐步抽象和解耦的設計思想。這種思想在C#等面向對象的編 ...
  • 關於Python中的私有屬性和私有方法 Python對於類的成員沒有嚴格的訪問控制限制,這與其他面相對對象語言有區別。關於私有屬性和私有方法,有如下要點: 1、通常我們約定,兩個下劃線開頭的屬性是私有的(private)。其他為公共的(public); 2、類內部可以訪問私有屬性(方法); 3、類外 ...
  • C++ 訪問說明符 訪問說明符是 C++ 中控制類成員(屬性和方法)可訪問性的關鍵字。它們用於封裝類數據並保護其免受意外修改或濫用。 三種訪問說明符: public:允許從類外部的任何地方訪問成員。 private:僅允許在類內部訪問成員。 protected:允許在類內部及其派生類中訪問成員。 示 ...
  • 寫這個隨筆說一下C++的static_cast和dynamic_cast用在子類與父類的指針轉換時的一些事宜。首先,【static_cast,dynamic_cast】【父類指針,子類指針】,兩兩一組,共有4種組合:用 static_cast 父類轉子類、用 static_cast 子類轉父類、使用 ...
  • /******************************************************************************************************** * * * 設計雙向鏈表的介面 * * * * Copyright (c) 2023-2 ...
  • 相信接觸過spring做開發的小伙伴們一定使用過@ComponentScan註解 @ComponentScan("com.wangm.lifecycle") public class AppConfig { } @ComponentScan指定basePackage,將包下的類按照一定規則註冊成Be ...
  • 操作系統 :CentOS 7.6_x64 opensips版本: 2.4.9 python版本:2.7.5 python作為腳本語言,使用起來很方便,查了下opensips的文檔,支持使用python腳本寫邏輯代碼。今天整理下CentOS7環境下opensips2.4.9的python模塊筆記及使用 ...