亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java線(xiàn)程通訊的實(shí)現(xiàn)方法總結(jié)

 更新時(shí)間:2023年05月26日 11:39:35   作者:javacn  
線(xiàn)程通訊指的是多個(gè)線(xiàn)程之間通過(guò)共享內(nèi)存或消息傳遞等方式來(lái)協(xié)調(diào)和同步它們的執(zhí)行,線(xiàn)程通訊的實(shí)現(xiàn)方式主要有以下兩種:共享內(nèi)存和消息傳遞,本文詳細(xì)介紹了Java線(xiàn)程是如何通訊的,感興趣的同學(xué)可以參考閱讀

線(xiàn)程通訊指的是多個(gè)線(xiàn)程之間通過(guò)共享內(nèi)存或消息傳遞等方式來(lái)協(xié)調(diào)和同步它們的執(zhí)行。在多線(xiàn)程編程中,通常會(huì)出現(xiàn)多個(gè)線(xiàn)程需要共同完成某個(gè)任務(wù)的情況,這時(shí)就需要線(xiàn)程之間進(jìn)行通訊,以保證任務(wù)能夠順利地執(zhí)行。

線(xiàn)程通訊的實(shí)現(xiàn)方式主要有以下兩種:

  • 共享內(nèi)存:多個(gè)線(xiàn)程可以訪(fǎng)問(wèn)同一個(gè)共享內(nèi)存區(qū)域,通過(guò)讀取和寫(xiě)入內(nèi)存中的數(shù)據(jù)來(lái)進(jìn)行通訊和同步。

  • 消息傳遞:多個(gè)線(xiàn)程之間通過(guò)消息隊(duì)列、管道、信號(hào)量等機(jī)制來(lái)傳遞信息和同步狀態(tài)。

常見(jiàn)場(chǎng)景

線(xiàn)程通訊的常見(jiàn)場(chǎng)景有以下幾個(gè):

  • 多個(gè)線(xiàn)程共同完成某個(gè)任務(wù):例如一個(gè)爬蟲(chóng)程序需要多個(gè)線(xiàn)程同時(shí)抓取不同的網(wǎng)頁(yè),然后將抓取結(jié)果合并保存到數(shù)據(jù)庫(kù)中。這時(shí)需要線(xiàn)程通訊來(lái)協(xié)調(diào)各個(gè)線(xiàn)程的執(zhí)行順序和共享數(shù)據(jù)。

  • 避免資源沖突:多個(gè)線(xiàn)程訪(fǎng)問(wèn)共享資源時(shí)可能會(huì)引發(fā)競(jìng)爭(zhēng)條件,例如多個(gè)線(xiàn)程同時(shí)讀寫(xiě)一個(gè)文件或數(shù)據(jù)庫(kù)。這時(shí)需要線(xiàn)程通訊來(lái)同步線(xiàn)程之間的數(shù)據(jù)訪(fǎng)問(wèn),避免資源沖突。

  • 保證順序執(zhí)行:在某些情況下,需要保證多個(gè)線(xiàn)程按照一定的順序執(zhí)行,例如一個(gè)多線(xiàn)程排序算法。這時(shí)需要線(xiàn)程通訊來(lái)協(xié)調(diào)各個(gè)線(xiàn)程的執(zhí)行順序。

  • 線(xiàn)程之間的互斥和同步:有些場(chǎng)景需要確保只有一個(gè)線(xiàn)程能夠訪(fǎng)問(wèn)某個(gè)共享資源,例如一個(gè)計(jì)數(shù)器。這時(shí)需要使用線(xiàn)程通訊機(jī)制來(lái)實(shí)現(xiàn)線(xiàn)程之間的互斥和同步。

實(shí)現(xiàn)方法

線(xiàn)程通訊的實(shí)現(xiàn)方法有以下幾種:

  • 等待和通知機(jī)制:使用 Object 類(lèi)的 wait() 和 notify() 方法來(lái)實(shí)現(xiàn)線(xiàn)程之間的通訊。當(dāng)一個(gè)線(xiàn)程需要等待另一個(gè)線(xiàn)程執(zhí)行完某個(gè)操作時(shí),它可以調(diào)用 wait() 方法使自己進(jìn)入等待狀態(tài),同時(shí)釋放占有的鎖,等待其他線(xiàn)程調(diào)用 notify() 或 notifyAll() 方法來(lái)喚醒它。被喚醒的線(xiàn)程會(huì)重新嘗試獲取鎖并繼續(xù)執(zhí)行。

  • 信號(hào)量機(jī)制:使用 Java 中的 Semaphore 類(lèi)來(lái)實(shí)現(xiàn)線(xiàn)程之間的同步和互斥。Semaphore 是一個(gè)計(jì)數(shù)器,用來(lái)控制同時(shí)訪(fǎng)問(wèn)某個(gè)資源的線(xiàn)程數(shù)。當(dāng)某個(gè)線(xiàn)程需要訪(fǎng)問(wèn)共享資源時(shí),它必須先從 Semaphore 中獲取一個(gè)許可證,如果已經(jīng)沒(méi)有許可證可用,線(xiàn)程就會(huì)被阻塞,直到其他線(xiàn)程釋放了許可證。

  • 柵欄機(jī)制:使用 Java 中的 CyclicBarrier 類(lèi)來(lái)實(shí)現(xiàn)多個(gè)線(xiàn)程之間的同步,它允許多個(gè)線(xiàn)程在指定的屏障處等待,并在所有線(xiàn)程都達(dá)到屏障時(shí)繼續(xù)執(zhí)行。

  • 鎖機(jī)制:使用 Java 中的 Lock 接口和 Condition 接口來(lái)實(shí)現(xiàn)線(xiàn)程之間的同步和互斥。Lock 是一種更高級(jí)的互斥機(jī)制,它允許多個(gè)條件變量(Condition)并支持在同一個(gè)鎖上等待和喚醒。

具體代碼實(shí)現(xiàn)

等待通知實(shí)現(xiàn)

以下是一個(gè)簡(jiǎn)單的 wait() 和 notify() 方法的等待通知示例:

public class WaitNotifyDemo {
public static void main(String[] args) {
        Object lock = new Object();
        ThreadA threadA = new ThreadA(lock);
        ThreadB threadB = new ThreadB(lock);
        threadA.start();
        threadB.start();
}
static class ThreadA extends Thread {
        private Object lock;
        public ThreadA(Object lock) {
            this.lock = lock;
        }
        public void run() {
            synchronized (lock) {
                System.out.println("ThreadA start...");
                try {
                    lock.wait(); // 線(xiàn)程A等待
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("ThreadA end...");
            }
        }
    }
static class ThreadB extends Thread {
        private Object lock;
        public ThreadB(Object lock) {
            this.lock = lock;
        }
        public void run() {
            synchronized (lock) {
                System.out.println("ThreadB start...");
                lock.notify(); // 喚醒線(xiàn)程A
                System.out.println("ThreadB end...");
            }
        }
    }
   }

在這個(gè)示例中,定義了一個(gè)共享對(duì)象 lock,ThreadA 線(xiàn)程先獲取 lock 鎖,并調(diào)用 lock.wait() 方法進(jìn)入等待狀態(tài)。ThreadB 線(xiàn)程在獲取 lock 鎖之后,調(diào)用 lock.notify() 方法喚醒 ThreadA 線(xiàn)程,然后 ThreadB 線(xiàn)程執(zhí)行完畢。

運(yùn)行以上程序的執(zhí)行結(jié)果如下:

ThreadA start...

ThreadB start...

ThreadB end...

ThreadA end...

信號(hào)量實(shí)現(xiàn)

在 Java 中使用 Semaphore 實(shí)現(xiàn)信號(hào)量,Semaphore 是一個(gè)計(jì)數(shù)器,用來(lái)控制同時(shí)訪(fǎng)問(wèn)某個(gè)資源的線(xiàn)程數(shù)。當(dāng)某個(gè)線(xiàn)程需要訪(fǎng)問(wèn)共享資源時(shí),它必須先從 Semaphore 中獲取一個(gè)許可證,如果已經(jīng)沒(méi)有許可證可用,線(xiàn)程就會(huì)被阻塞,直到其他線(xiàn)程釋放了許可證。它的示例代碼如下:

import java.util.concurrent.Semaphore;
public class SemaphoreDemo {
    public static void main(String[] args) {
        Semaphore semaphore = new Semaphore(2);
        for (int i = 0; i < 5; i++) {
            new Thread(new Worker(i, semaphore)).start();
        }
    }
    static class Worker implements Runnable {
        private int id;
        private Semaphore semaphore;
        public Worker(int id, Semaphore semaphore) {
            this.id = id;
            this.semaphore = semaphore;
        }
        @Override
        public void run() {
            try {
                semaphore.acquire();
                System.out.println("Worker " + id + " acquired permit.");
                Thread.sleep(1000);
                System.out.println("Worker " + id + " released permit.");
                semaphore.release();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

在這個(gè)示例中,創(chuàng)建了一個(gè) Semaphore 對(duì)象,并且設(shè)置了許可數(shù)為 2。然后創(chuàng)建了 5 個(gè) Worker 線(xiàn)程,每個(gè) Worker 線(xiàn)程需要獲取 Semaphore 的許可才能執(zhí)行任務(wù)。每個(gè) Worker 線(xiàn)程在執(zhí)行任務(wù)之前先調(diào)用 semaphore.acquire() 方法獲取許可,如果沒(méi)有許可則會(huì)阻塞,直到 Semaphore 釋放許可。執(zhí)行完任務(wù)之后調(diào)用 semaphore.release() 方法釋放許可。 運(yùn)行以上程序的執(zhí)行結(jié)果如下:

Worker 0 acquired permit.

Worker 1 acquired permit.

Worker 1 released permit.

Worker 0 released permit.

Worker 2 acquired permit.

Worker 3 acquired permit.

Worker 2 released permit.

Worker 4 acquired permit.

Worker 3 released permit.

Worker 4 released permit.

柵欄實(shí)現(xiàn)

在 Java 中,可以使用 CyclicBarrier 或 CountDownLatch 來(lái)實(shí)現(xiàn)線(xiàn)程的同步,它們兩個(gè)使用類(lèi)似,接下來(lái)我們就是 CyclicBarrier 來(lái)演示一下線(xiàn)程的同步,CyclicBarrier 的示例代碼如下:

import java.util.concurrent.BrokenBarrierException;
import java.util.concurrent.CyclicBarrier;
public class CyclicBarrierDemo {
    public static void main(String[] args) {
        CyclicBarrier cyclicBarrier = new CyclicBarrier(3, new Runnable() {
            @Override
            public void run() {
                System.out.println("All threads have reached the barrier.");
            }
        });
        for (int i = 1; i <= 3; i++) {
            new Thread(new Worker(i, cyclicBarrier)).start();
        }
    }
    static class Worker implements Runnable {
        private int id;
        private CyclicBarrier cyclicBarrier;
        public Worker(int id, CyclicBarrier cyclicBarrier) {
            this.id = id;
            this.cyclicBarrier = cyclicBarrier;
        }
        @Override
        public void run() {
            try {
                System.out.println("Worker " + id + " is working.");
                Thread.sleep((long) (Math.random() * 2000));
                System.out.println("Worker " + id + " has reached the barrier.");
                cyclicBarrier.await();
                System.out.println("Worker " + id + " is continuing the work.");
            } catch (InterruptedException | BrokenBarrierException e) {
                e.printStackTrace();
            }
        }
    }
}

在這個(gè)示例中,創(chuàng)建了一個(gè) CyclicBarrier 對(duì)象,并且設(shè)置了參與線(xiàn)程數(shù)為 3。然后創(chuàng)建了 3 個(gè) Worker 線(xiàn)程,每個(gè) Worker 線(xiàn)程會(huì)先執(zhí)行一些任務(wù),然后等待其他線(xiàn)程到達(dá) Barrier。所有線(xiàn)程都到達(dá) Barrier 之后,Barrier 會(huì)釋放所有線(xiàn)程并執(zhí)行設(shè)置的 Runnable 任務(wù)。 運(yùn)行以上程序的執(zhí)行結(jié)果如下:

Worker 2 is working.

Worker 3 is working.

Worker 1 is working.

Worker 3 has reached the barrier.

Worker 1 has reached the barrier.

Worker 2 has reached the barrier.

All threads have reached the barrier.

Worker 2 is continuing the work.

Worker 3 is continuing the work.

Worker 1 is continuing the work.

從以上執(zhí)行結(jié)果可以看出,CyclicBarrier 保證了所有 Worker 線(xiàn)程都到達(dá) Barrier 之后才能繼續(xù)執(zhí)行后面的任務(wù),這樣可以保證線(xiàn)程之間的同步和協(xié)作。在本示例中,所有線(xiàn)程都在 Barrier 處等待了一段時(shí)間,等所有線(xiàn)程都到達(dá) Barrier 之后才繼續(xù)執(zhí)行后面的任務(wù)。

鎖機(jī)制實(shí)現(xiàn)

以下是一個(gè)使用 Condition 的示例:

import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
public class ConditionDemo {
    private Lock lock = new ReentrantLock();
    private Condition condition = lock.newCondition();
    private volatile boolean flag = false;
    public static void main(String[] args) {
        ConditionDemo demo = new ConditionDemo();
        new Thread(demo::waitCondition).start();
        new Thread(demo::signalCondition).start();
    }
    private void waitCondition() {
        lock.lock();
        try {
            while (!flag) {
                System.out.println(Thread.currentThread().getName() + " is waiting for signal.");
                condition.await();
            }
            System.out.println(Thread.currentThread().getName() + " received signal.");
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            lock.unlock();
        }
    }
    private void signalCondition() {
        lock.lock();
        try {
            Thread.sleep(3000); // 模擬等待一段時(shí)間后發(fā)送信號(hào)
            flag = true;
            System.out.println(Thread.currentThread().getName() + " sends signal.");
            condition.signalAll();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            lock.unlock();
        }
    }
}

在這個(gè)示例中,創(chuàng)建了一個(gè) Condition 對(duì)象和一個(gè) Lock 對(duì)象,然后創(chuàng)建了兩個(gè)線(xiàn)程,一個(gè)線(xiàn)程等待 Condition 信號(hào),另一個(gè)線(xiàn)程發(fā)送 Condition 信號(hào)。

等待線(xiàn)程在獲得鎖后,判斷標(biāo)志位是否為 true,如果為 false,則等待 Condition 信號(hào);如果為 true,則繼續(xù)執(zhí)行后面的任務(wù)。

發(fā)送線(xiàn)程在獲得鎖后,等待一段時(shí)間后,將標(biāo)志位設(shè)置為 true,并且發(fā)送 Condition 信號(hào)。 運(yùn)行以上程序的執(zhí)行結(jié)果如下:

Thread-0 is waiting for signal.

Thread-1 sends signal.

Thread-0 received signal.

從上面執(zhí)行結(jié)果可以看出,等待線(xiàn)程在等待 Condition 信號(hào)的時(shí)候被阻塞,直到發(fā)送線(xiàn)程發(fā)送了 Condition 信號(hào),等待線(xiàn)程才繼續(xù)執(zhí)行后面的任務(wù)。Condition 對(duì)象提供了一種更加靈活的線(xiàn)程通信方式,可以精確地控制線(xiàn)程的等待和喚醒。

小結(jié)

線(xiàn)程通訊指的是多個(gè)線(xiàn)程之間通過(guò)共享內(nèi)存或消息傳遞等方式來(lái)協(xié)調(diào)和同步它們的執(zhí)行,它的實(shí)現(xiàn)方法有很多:比如 wait() 和 notify() 的等待和通知機(jī)制、Semaphore 信號(hào)量機(jī)制、CyclicBarrier 柵欄機(jī)制,以及 Condition 的鎖機(jī)制等。

以上就是Java線(xiàn)程通訊的實(shí)現(xiàn)方式總結(jié)的詳細(xì)內(nèi)容,更多關(guān)于Java線(xiàn)程通訊的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Java生成短8位UUID的實(shí)現(xiàn)方案

    Java生成短8位UUID的實(shí)現(xiàn)方案

    在Java中,UUID通常用于生成全局唯一的標(biāo)識(shí)符,標(biāo)準(zhǔn)的UUID是128位的,由32個(gè)十六進(jìn)制數(shù)字組成,并通過(guò)特定的算法保證其在全球范圍內(nèi)的唯一性,本文給大家介紹了一個(gè)簡(jiǎn)單的Java方法,用于生成一個(gè)較短的8位UUID,需要的朋友可以參考下
    2025-01-01
  • springcloud結(jié)合bytetcc實(shí)現(xiàn)數(shù)據(jù)強(qiáng)一致性原理解析

    springcloud結(jié)合bytetcc實(shí)現(xiàn)數(shù)據(jù)強(qiáng)一致性原理解析

    這篇文章主要介紹了springcloud結(jié)合bytetcc實(shí)現(xiàn)數(shù)據(jù)強(qiáng)一致性原理解析,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-03-03
  • 使用Lucene實(shí)現(xiàn)一個(gè)簡(jiǎn)單的布爾搜索功能

    使用Lucene實(shí)現(xiàn)一個(gè)簡(jiǎn)單的布爾搜索功能

    Lucene是一個(gè)全文搜索框架,而不是應(yīng)用產(chǎn)品。因此它并不像www.baidu.com 或者google Desktop那么拿來(lái)就能用,它只是提供了一種工具讓你能實(shí)現(xiàn)這些產(chǎn)品。接下來(lái)通過(guò)本文給大家介紹使用Lucene實(shí)現(xiàn)一個(gè)簡(jiǎn)單的布爾搜索功能
    2017-04-04
  • RocketMQ的順序消費(fèi)機(jī)制詳解

    RocketMQ的順序消費(fèi)機(jī)制詳解

    這篇文章主要介紹了RocketMQ的順序消費(fèi)機(jī)制詳解,順序消息是指對(duì)于一個(gè)指定的?Topic?,消息嚴(yán)格按照先進(jìn)先出(FIFO)的原則進(jìn)行消息發(fā)布和消費(fèi),即先發(fā)布的消息先消費(fèi),后發(fā)布的消息后消費(fèi),,需要的朋友可以參考下
    2023-10-10
  • 關(guān)于List、Map、Stream初始化方式

    關(guān)于List、Map、Stream初始化方式

    這篇文章主要介紹了關(guān)于List、Map、Stream初始化方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-08-08
  • SpringBoot打包發(fā)布到linux上(centos 7)的步驟

    SpringBoot打包發(fā)布到linux上(centos 7)的步驟

    這篇文章主要介紹了SpringBoot打包發(fā)布到linux上(centos 7)的步驟,幫助大家更好的理解和使用springboot框架,感興趣的朋友可以了解下
    2020-12-12
  • 你知道怎么從Python角度學(xué)習(xí)Java基礎(chǔ)

    你知道怎么從Python角度學(xué)習(xí)Java基礎(chǔ)

    這篇文章主要為大家詳細(xì)介紹了Python角度學(xué)習(xí)Java基礎(chǔ)的方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助
    2022-02-02
  • 在RedHat系統(tǒng)上安裝JDK與Tomcat的步驟

    在RedHat系統(tǒng)上安裝JDK與Tomcat的步驟

    這篇文章主要介紹了在RedHat系統(tǒng)上安裝Java與Tomcat的步驟,同樣適用于CentOS等RedHat系的Linux系統(tǒng),需要的朋友可以參考下
    2015-11-11
  • Spring Security實(shí)現(xiàn)驗(yàn)證碼登錄功能

    Spring Security實(shí)現(xiàn)驗(yàn)證碼登錄功能

    這篇文章主要介紹了Spring Security實(shí)現(xiàn)驗(yàn)證碼登錄功能,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-01-01
  • Spring Cloud Stream如何實(shí)現(xiàn)服務(wù)之間的通訊

    Spring Cloud Stream如何實(shí)現(xiàn)服務(wù)之間的通訊

    這篇文章主要介紹了Spring Cloud Stream如何實(shí)現(xiàn)服務(wù)之間的通訊,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-10-10

最新評(píng)論