国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

Java并發編程之線程間通訊(下)-生產者與消費者

lufficc / 2709人閱讀

摘要:前文回顧上一篇文章重點嘮叨了中協調線程間通信的機制,它有力的保證了線程間通信的安全性以及便利性。所以同一時刻廚師線程和服務員線程不會同時在等待隊列中。對于在操作系統中線程的阻塞狀態,語言中用和這三個狀態分別表示。

前文回顧

上一篇文章重點嘮叨了java中協調線程間通信的wait/notify機制,它有力的保證了線程間通信的安全性以及便利性。本篇將介紹wait/notify機制的一個應用以及更多線程間通信的內容。

生產者-消費者模式

目光從廁所轉到飯館,一個飯館里通常都有好多廚師以及好多服務員,這里我們把廚師稱為生產者,把服務員稱為消費者,廚師和服務員是不直接打交道的,而是在廚師做好菜之后放到窗口,服務員從窗口直接把菜端走給客人就好了,這樣會極大的提升工作效率,因為省去了生產者和消費者之間的溝通成本。從java的角度看這個事情,每一個廚師就相當于一個生產者線程,每一個服務員都相當于一個消費者線程,而放菜的窗口就相當于一個緩沖隊列生產者線程不斷把生產好的東西放到緩沖隊列里,消費者線程不斷從緩沖隊列里取東西,畫個圖就像是這樣:

現實中放菜的窗口能放的菜數量是有限的,我們假設這個窗口只能放5個菜。那么廚師在做完菜之后需要看一下窗口是不是滿了,如果窗口已經滿了的話,就在一旁抽根煙等待,直到有服務員來取菜的時候通知一下廚師窗口有了空閑,可以放菜了,這時廚師再把自己做的菜放到窗口上去炒下一個菜。從服務員的角度來說,如果窗口是空的,那么也去一旁抽根煙等待,直到有廚師把菜做好了放到窗口上,并且通知他們一下,然后再把菜端走。

我們先用java抽象一下菜:

public class Food {

    private static int counter = 0;

    private int i;  //代表生產的第幾個菜

    public Food() {
        i = ++counter;
    }

    @Override
    public String toString() {
        return "第" + i + "個菜";
    }
}

每次創建Food對象,字段i的值都會加1,代表這是創建的第幾道菜。

為了故事的順利進行,我們首先定義一個工具類:

class SleepUtil {

    private static Random random = new Random();

    public static void randomSleep() {
        try {
            Thread.sleep(random.nextInt(1000));
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
    }
}

SleepUtil的靜態方法randomSleep代表當前線程隨機休眠一秒內的時間。

然后我們再用java定義一下廚師:

public class Cook extends Thread {

    private Queue queue;

    public Cook(Queue queue, String name) {
        super(name);
        this.queue = queue;
    }

    @Override
    public void run() {

        while (true) {
            SleepUtil.randomSleep();    //模擬廚師炒菜時間

            Food food = new Food();
            System.out.println(getName() + " 生產了" + food);

            synchronized (queue) {
                while (queue.size() > 4) {
                    try {
                        System.out.println("隊列元素超過5個,為:" + queue.size() + " " + getName() + "抽根煙等待中");
                        queue.wait();

                    } catch (InterruptedException e) {
                        throw new RuntimeException(e);
                    }
                }

                queue.add(food);
                queue.notifyAll();
            }
        }
    }
}

我們說每一個廚師Cook都是一個線程,內部維護了一個名叫queue的隊列。在run方法中是一個死循環,代表不斷的生產Food。他每生產一個Food后,都要判斷queue隊列中元素的個數是不是大于4,如果大于4的話,就調用queue.wait()等待,如果不大于4的話,就把創建號的Food對象放到queue隊列中,由于可能多個線程同時訪問queue的各個方法,所以對這段代碼用queue對象來加鎖保護。當向隊列添加完剛創建的Food對象之后,就可以通知queue這個鎖對象關聯的等待隊列中的服務員線程們可以繼續端菜了。

然后我們再用java定義一下服務員:

class Waiter extends Thread {

    private Queue queue;

    public Waiter(Queue queue, String name) {
        super(name);
        this.queue = queue;
    }

    @Override
    public void run() {

        while (true) {

            Food food;
            synchronized (queue) {
                while (queue.size() < 1) {
                    try {
                        System.out.println("隊列元素個數為: " + queue.size() + "," + getName() + "抽根煙等待中");
                        queue.wait();
                    } catch (InterruptedException e) {
                        throw new RuntimeException(e);
                    }
                }

                food = queue.remove();
                System.out.println(getName() + " 獲取到:" + food);
                queue.notifyAll();
            }

            SleepUtil.randomSleep();    //模擬服務員端菜時間
        }
    }
}

每個服務員也是一個線程,和廚師一樣,都在內部維護了一個名叫queue的隊列。在run方法中是一個死循環,代表不斷的從隊列中取走Food。每次在從queue隊列中取Food對象的時候,都需要判斷一下隊列中的元素是否小于1,如果小于1的話,就調用queue.wait()等待,如果不小于1的話,也就是隊列里有元素,就從隊列里取走一個Food對象,并且通知與queue這個鎖對象關聯的等待隊列中的廚師線程們可以繼續向隊列里放入Food對象了。

在廚師和服務員線程類都定義好了之后,我們再創建一個Restaurant類,來看看在餐館里真實發生的事情:

public class Restaurant {

    public static void main(String[] args) {

        Queue queue = new LinkedList<>();
        new Cook(queue, "1號廚師").start();
        new Cook(queue, "2號廚師").start();
        new Cook(queue, "3號廚師").start();
        new Waiter(queue, "1號服務員").start();
        new Waiter(queue, "2號服務員").start();
        new Waiter(queue, "3號服務員").start();
    }
}

我們在Restaurant中安排了3個廚師和3個服務員,大家執行一下這個程序,會發現在如果廚師生產的過快,廚師就會等待,如果服務員端菜速度過快,服務員就會等待。但是整個過程廚師和服務員是沒有任何關系的,它們是通過隊列queue實現了所謂的解耦。

這個過程雖然不是很復雜,但是使用中還是需要注意一些問題:

我們這里的廚師和服務員使用同一個鎖queue

使用同一個鎖是因為對queue的操作只能用同一個鎖來保護,假設使用不同的鎖,廚師線程調用queue.add方法,服務員線程調用queue.remove方法,這兩個方法都不是原子操作,多線程并發執行的時候會出現不可預測的結果,所以我們使用同一個鎖來保護對queue這個變量的操作,這一點我們在嘮叨設計線程安全類的時候已經強調過了。

廚師和服務員線程使用同一個鎖queue的后果就是廚師線程和服務員線程使用的是同一個等待隊列。

但是同一時刻廚師線程和服務員線程不會同時在等待隊列中,因為當廚師線程在wait的時候,隊列里的元素肯定是5,此時服務員線程肯定是不會wait的,但是消費的過程是被鎖對象queue保護的,所以在一個服務員線程消費了一個Food之后,就會調用notifyAll來喚醒等待隊列中的廚師線程們;當消費者線程在wait的時候,隊列里的元素肯定是0,此時廚師線程肯定是不會wait的,生產的過程是被鎖對象queue保護的,所以在一個廚師線程生產了一個Food對象之后,就會調用notifyAll來喚醒等待隊列中的服務員線程們。所以同一時刻廚師線程服務員線程不會同時在等待隊列中。

在生產和消費過程,我們都調用了SleepUtil.randomSleep();。

我們這里的生產者-消費者模型是把實際使用的場景進行了簡化,真正的實際場景中生產過程和消費過程一般都會很耗時,這些耗時的操作最好不要放在同步代碼塊中,這樣會造成別的線程的長時間阻塞。如果把生產過程和消費過程都放在同步代碼塊中,也就是說在一個廚師炒菜的同時不允許別的廚師炒菜,在一個服務員端菜的同時不允許別的程序員端菜,這個顯然是不合理的,大家需要注意這一點。

以上就是wait/notify機制的一個現實應用:生產者-消費者模式的一個簡介。

管道輸入/輸出流

還記得在嘮叨I/O的時候提到的管道流么,這些管道流就是用于在不同線程之間的數據傳輸,一共有四種管道流:

PipedInputStream:管道輸入字節流

PipedOutputStream:管道輸出字節流

PipedReader:管道輸入字符流

PipedWriter:管道輸出字符流

字節流和字符流的用法是差不多的,我們下邊以字節流為例來嘮叨一下管道流的用法。

一個線程可以持有一個PipedInputStream對象,這個PipedInputStream對象在內部維護了一個字節數組,默認大小為1024字節。它并不能多帶帶使用,需要與另一個線程持有的一個PipedOutputStream建立關聯,PipedOutputStream往該字節數組中寫數據,PipedInputStream從該字節數組中讀數據,從而實現兩個線程的通信。

PipedInputStream

先看一下它的幾個構造方法:


它有一個特別重要的方法就是:

PipedOutputStream

看一下它的構造方法:

它也有一個連接到管道輸入流的方法:

使用示例

管道流的通常使用場景就是一個線程持有一個PipedInputStream對象,另一個線程持有一個PipedOutputStream對象,然后把這兩個輸入輸出管道流通過connect方法建立連接,此后從管道輸出流寫入的數據就可以通過管道輸入流讀出,從而實現了兩個線程間的數據交換,也就是實現了線程間的通信

public class PipedDemo {

    public static void main(String[] args){

        PipedInputStream in = new PipedInputStream();
        PipedOutputStream out = new PipedOutputStream();

        try {
            in.connect(out);    //將輸入流和輸出流建立關聯
        } catch (IOException e) {
            throw new RuntimeException(e);
        }

        new ReadThread(in).start();
        new WriteThread(out).start();
    }
}

class ReadThread extends Thread {

    private PipedInputStream in;

    public ReadThread(PipedInputStream in) {
        this.in = in;
    }

    @Override
    public void run() {

        int i = 0;
        try {
            while ((i=in.read()) != -1) {   //從輸入流讀取數據
                System.out.println(i);
            }
        } catch (IOException e) {
            throw new RuntimeException(e);
        } finally {
            try {
                in.close();
            } catch (IOException e) {
                throw new RuntimeException(e);
            }
        }
    }
}

class WriteThread extends Thread {

    private PipedOutputStream out;

    public WriteThread(PipedOutputStream out) {
        this.out = out;
    }

    @Override
    public void run() {
        byte[] bytes = {1, 2, 3, 4, 5};
        try {
            out.write(bytes);   //向輸出流寫入數據
            out.flush();
        } catch (IOException e) {
            throw new RuntimeException(e);
        } finally {
            try {
                out.close();
            } catch (IOException e) {
                throw new RuntimeException(e);
            }
        }
    }
}

執行結果是:

1
2
3
4
5
join方法

我們前邊說過這個方法,比如有代碼是這樣:

public static void main(String[] args) {

    Thread t = new Thread(new Runnable() {

        @Override
        public void run() {
            // ... 線程t執行的具體任務
        }
    }, "t");

    t.start();

    t.join();
    System.out.println("t線程執行完了,繼續執行main線程");
}

main線程中調用t.join(),代表main線程需要等待t線程執行完成后才能繼續執行。也就是說,這個join方法可以協調各個線程之間的執行順序。它的實現其實很簡單:

public final synchronized void join() throws InterruptedException {
    while (isAlive()) {
        wait();
    }
}

需要注意的是,join方法Thread類的成員方法。上邊例子中在main線程中調用t.join()的意思就是,使用Thread對象t作為鎖對象,如果t線程還活著,就調用wait(),把main線程放到與t對象關聯的等待隊列里,直到t線程執行結束,系統會主動調用一下t.notifyAll(),把與t對象關聯的等待隊列中的線程全部移出,從而main線程可以繼續執行~

當然它還有兩個指定等待時間的重載方法:

java線程的狀態

java為了方便的管理線程,對底層的操作系統的線程狀態做了一些抽象封裝,定義了如下的線程狀態:

需要注意的是:

對于在操作系統中線程的運行/就緒狀態,java語言中統一用RUNNABLE狀態來表示。

對于在操作系統中線程的阻塞狀態,java語言中用BLOCKEDWAITINGTIME_WAITING這三個狀態分別表示。

也就是對阻塞狀態進行了進一步細分。對于因為獲取不到鎖而產生的阻塞稱為BLOCKED狀態,因為調用wait或者join方法而產生的阻塞稱為WAITING狀態,因為調用有超時時間的waitjoin或者sleep方法而產生的在有限時間內阻塞稱為TIME_WAITING狀態。

大家可以通過這個圖來詳細的看一下各個狀態之間的轉換過程:

java這么劃分線程的狀態純屬于方便自己的管理,比如它會給在WAITINGTIMED_WAITING狀態的線程分別建立不同的隊列,來方便實施不同的恢復策略~所以大家也不用糾結為啥和操作系統中定義的不一樣,其實操作系統中對各個狀態的線程仍然有各種細分來方便管理,如果是你去設計一個語言或者一個操作系統,你也可以為了自己的方便來定義一下線程的各種狀態。我們作為語言的使用者,首先還是把這些狀態記住了再說哈

文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。

轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/74171.html

相關文章

  • Javag工程師成神路(2019正式版)

    摘要:結構型模式適配器模式橋接模式裝飾模式組合模式外觀模式享元模式代理模式。行為型模式模版方法模式命令模式迭代器模式觀察者模式中介者模式備忘錄模式解釋器模式模式狀態模式策略模式職責鏈模式責任鏈模式訪問者模式。 主要版本 更新時間 備注 v1.0 2015-08-01 首次發布 v1.1 2018-03-12 增加新技術知識、完善知識體系 v2.0 2019-02-19 結構...

    Olivia 評論0 收藏0
  • python并發4:使用thread處理并發

    摘要:如果某線程并未使用很多操作,它會在自己的時間片內一直占用處理器和。在中使用線程在和等大多數類系統上運行時,支持多線程編程。守護線程另一個避免使用模塊的原因是,它不支持守護線程。 這一篇是Python并發的第四篇,主要介紹進程和線程的定義,Python線程和全局解釋器鎖以及Python如何使用thread模塊處理并發 引言&動機 考慮一下這個場景,我們有10000條數據需要處理,處理每條...

    joywek 評論0 收藏0
  • Java并發編程筆記(二)

    摘要:本文探討并發中的其它問題線程安全可見性活躍性等等。當閉鎖到達結束狀態時,門打開并允許所有線程通過。在從返回時被叫醒時,線程被放入鎖池,與其他線程競爭重新獲得鎖。 本文探討Java并發中的其它問題:線程安全、可見性、活躍性等等。 在行文之前,我想先推薦以下兩份資料,質量很高:極客學院-Java并發編程讀書筆記-《Java并發編程實戰》 線程安全 《Java并發編程實戰》中提到了太多的術語...

    NickZhou 評論0 收藏0
  • 分布式服務框架遠程通訊技術及原理分析

    摘要:微軟的雖然引入了事件機制,可以在隊列收到消息時觸發事件,通知訂閱者。由微軟作為主要貢獻者的,則對以及做了進一層包裝,并能夠很好地實現這一模式。 在分布式服務框架中,一個最基礎的問題就是遠程服務是怎么通訊的,在Java領域中有很多可實現遠程通訊的技術,例如:RMI、MINA、ESB、Burlap、Hessian、SOAP、EJB和JMS等,這些名詞之間到底是些什么關系呢,它們背后到底是基...

    sorra 評論0 收藏0
  • 分布式服務框架遠程通訊技術及原理分析

    摘要:微軟的雖然引入了事件機制,可以在隊列收到消息時觸發事件,通知訂閱者。由微軟作為主要貢獻者的,則對以及做了進一層包裝,并能夠很好地實現這一模式。 在分布式服務框架中,一個最基礎的問題就是遠程服務是怎么通訊的,在Java領域中有很多可實現遠程通訊的技術,例如:RMI、MINA、ESB、Burlap、Hessian、SOAP、EJB和JMS等,這些名詞之間到底是些什么關系呢,它們背后到底是基...

    0xE7A38A 評論0 收藏0

發表評論

0條評論

lufficc

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<