Java中notify()的使用

在Java中,線程同步與互斥是非常重要的概念。以往我們使用synchronized關鍵字來鎖定某個方法或某段代碼塊,以保持線程的同步與互斥。在這篇文章中,我們將介紹另一個重要的關鍵字——notify()。notify()方法的作用是喚醒等待線程隊列中的某個線程,並將該線程移動到鎖池中。對於正在運行的線程,notify()方法不會產生任何影響。如果你想在某個時刻喚醒一個等待線程,你需要在代碼中仔細地實現它。

一、notify()方法的使用

我們將具體了解notify()方法的使用。notify()方法要在synchronized塊中調用,以確保該對象的鎖定機制。

public class NotifyTest {
     public static void main(String[] args) {
        ThreadB b = new ThreadB();
        b.start();
        
        synchronized(b) {
            try {
                System.out.println("Waiting for b to complete...");
                b.wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            
            System.out.println("Total is: " + b.total);
        }
     }
}

class ThreadB extends Thread {
    int total;
    
    public void run() {
        synchronized(this) {
            for(int i=0; i<100 ; i++) {
                total += i;
            }
            notify();
        }
    }
}

在這個例子中,我們創建了兩個線程:一個線程是主線程,另一個線程是ThreadB。main線程在運行時,調用synchronized方法鎖定ThreadB對象。然後,main線程調用ThreadB.wait()方法來等待ThreadB線程的完成。在ThreadB線程完成之後,total的值會被列印出來。

二、notify()方法的限制

notify()方法除了喚醒等待線程之外,它並沒有其他的效果。如果一個線程被notify()喚醒,它不會立即執行。相反,該線程需要重新進入該對象的鎖池並等待重新獲得該對象的鎖,然後才能繼續執行。

還有一個限制是,notify()方法只會喚醒等待隊列中的其中一個線程,並且我們不能指定哪個線程會被喚醒。也就是說,我們無法控制具體喚醒哪一個等待線程。

三、notifyAll()方法的使用

我們介紹了notify()方法的基本使用和限制,現在我們將介紹另一個方法:notifyAll()。notifyAll()方法喚醒等待隊列中的所有線程。

public class NotifyAllTest {
    public static void main(String[] args) {
        Message msg = new Message("process it");
        Waiter waiter1 = new Waiter(msg);
        new Thread(waiter1,"waiter1").start();
        
        Waiter waiter2 = new Waiter(msg);
        new Thread(waiter2, "waiter2").start();
        
        Notifier notifier = new Notifier(msg);
        new Thread(notifier, "notifier").start();
        System.out.println("All threads started");
    }
}

class Message {
    private String msg;

    public Message(String str){
        this.msg=str;
    }

    public String getMsg() {
        return msg;
    }

    public void setMsg(String str) {
        this.msg=str;
    }
}

class Waiter implements Runnable{
    private Message msg;

    public Waiter(Message m){
        this.msg=m;
    }

    public void run() {
        String name = Thread.currentThread().getName();
        synchronized (msg) {
            try{
                System.out.println(name+" waiting to get notified at time:"+System.currentTimeMillis());
                msg.wait();
            }catch(InterruptedException e){
                e.printStackTrace();
            }
            System.out.println(name+" waiter thread got notified at time:"+System.currentTimeMillis());
            //process the message now
            System.out.println(name+" processed: "+msg.getMsg());
        }
    }
}

class Notifier implements Runnable {
    private Message msg;

    public Notifier(Message msg) {
        this.msg = msg;
    }

    public void run() {
        Thread.currentThread().setName("notifier");
        System.out.println("Started notifying at time:"+System.currentTimeMillis());
        synchronized (msg) {
            msg.setMsg("Notifying all waiting threads");
            msg.notifyAll();
            // msg.notify();
        }
    }
}

在這個例子中,我們創建了一個Message類,一個Waiter類和一個Notifier類。Waiter類等待Notifier類的通知,Notifier類發出通知,等待的Waiter類會被喚醒並繼續執行。

在這個例子中,我們使用了wait()和notifyAll()方法。wait()方法可將當前線程放入對象的等待隊列中,並釋放對象鎖,以便其他線程可以獲得該對象的鎖並繼續執行。notifyAll()方法將喚醒所有正在等待該對象的線程,這些線程將重新進入對象的鎖池,並準備重新獲得對象的鎖。

四、總結

在本篇文章中,我們介紹了Java中notify()方法的基本使用和限制,以及notifyAll()方法的使用。notify()方法是喚醒等待隊列中的某個線程,notifyAll()方法是喚醒等待隊列中的所有線程。在使用notify()和notifyAll()方法時,需要確保已經獲得對象鎖,以便它可以執行正確的操作。

原創文章,作者:小藍,如若轉載,請註明出處:https://www.506064.com/zh-tw/n/193898.html

(0)
打賞 微信掃一掃 微信掃一掃 支付寶掃一掃 支付寶掃一掃
小藍的頭像小藍
上一篇 2024-12-02 09:41
下一篇 2024-12-02 09:41

相關推薦

  • java client.getacsresponse 編譯報錯解決方法

    java client.getacsresponse 編譯報錯是Java編程過程中常見的錯誤,常見的原因是代碼的語法錯誤、類庫依賴問題和編譯環境的配置問題。下面將從多個方面進行分析…

    編程 2025-04-29
  • Java JsonPath 效率優化指南

    本篇文章將深入探討Java JsonPath的效率問題,並提供一些優化方案。 一、JsonPath 簡介 JsonPath是一個可用於從JSON數據中獲取信息的庫。它提供了一種DS…

    編程 2025-04-29
  • Java騰訊雲音視頻對接

    本文旨在從多個方面詳細闡述Java騰訊雲音視頻對接,提供完整的代碼示例。 一、騰訊雲音視頻介紹 騰訊雲音視頻服務(Cloud Tencent Real-Time Communica…

    編程 2025-04-29
  • Java Bean載入過程

    Java Bean載入過程涉及到類載入器、反射機制和Java虛擬機的執行過程。在本文中,將從這三個方面詳細闡述Java Bean載入的過程。 一、類載入器 類載入器是Java虛擬機…

    編程 2025-04-29
  • Java Milvus SearchParam withoutFields用法介紹

    本文將詳細介紹Java Milvus SearchParam withoutFields的相關知識和用法。 一、什麼是Java Milvus SearchParam without…

    編程 2025-04-29
  • Java 8中某一周的周一

    Java 8是Java語言中的一個版本,於2014年3月18日發布。本文將從多個方面對Java 8中某一周的周一進行詳細的闡述。 一、數組處理 Java 8新特性之一是Stream…

    編程 2025-04-29
  • Java判斷字元串是否存在多個

    本文將從以下幾個方面詳細闡述如何使用Java判斷一個字元串中是否存在多個指定字元: 一、字元串遍歷 字元串是Java編程中非常重要的一種數據類型。要判斷字元串中是否存在多個指定字元…

    編程 2025-04-29
  • VSCode為什麼無法運行Java

    解答:VSCode無法運行Java是因為默認情況下,VSCode並沒有集成Java運行環境,需要手動添加Java運行環境或安裝相關插件才能實現Java代碼的編寫、調試和運行。 一、…

    編程 2025-04-29
  • Java任務下發回滾系統的設計與實現

    本文將介紹一個Java任務下發回滾系統的設計與實現。該系統可以用於執行複雜的任務,包括可回滾的任務,及時恢復任務失敗前的狀態。系統使用Java語言進行開發,可以支持多種類型的任務。…

    編程 2025-04-29
  • Java 8 Group By 會影響排序嗎?

    是的,Java 8中的Group By會對排序產生影響。本文將從多個方面探討Group By對排序的影響。 一、Group By的概述 Group By是SQL中的一種常見操作,它…

    編程 2025-04-29

發表回復

登錄後才能評論