手寫一個(gè)生產(chǎn)者--消費(fèi)者模型例子

在并發(fā)編程中砸脊,比較經(jīng)典的編程例子就是生產(chǎn)者和消費(fèi)者模型。下面就是一個(gè)例子來詮釋一下什么是生產(chǎn)者和消費(fèi)者以及他們的特點(diǎn)和注意點(diǎn)纬霞。

1凌埂、先定義一個(gè)數(shù)據(jù)對象,

public class Data {
    private String id;

    private String name;

    public Data(String id,String name){
        this.id = id;
        this.name = name;
    }
    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return "Data [id=" + id + ", name=" + name + "]";
    }
}

2.定義一個(gè)生產(chǎn)者诗芜,實(shí)現(xiàn)Runnable接口

public class Provider implements Runnable{
    //共享緩沖區(qū)
    private BlockingQueue<Data> queue;

    //多線程間是否啟動(dòng)變量瞳抓,有強(qiáng)制從主內(nèi)存中刷新的功能,及時(shí)返回線程狀態(tài)
    private volatile boolean isRunning = true;
    //id生成器
    private static AtomicInteger count = new AtomicInteger();

    //隨機(jī)對象
    private static Random r = new Random();

    public Provider(BlockingQueue queue){
        this.queue = queue;
    }

    @Override
    public void run() {
        while(isRunning){
            //隨機(jī)休眠0-1000毫秒 表示獲取數(shù)據(jù)
            try {
                Thread.sleep(r.nextInt(1000));
                //獲取的數(shù)據(jù)進(jìn)行累計(jì)
                int id  = count.incrementAndGet();
                //比如通過一個(gè)getData()方法獲取了
                Data data = new Data(Integer.toString(id),"數(shù)據(jù)"+id);
                System.out.println("當(dāng)前線程:"+ Thread.currentThread().getName() + ",獲取了數(shù)據(jù)伏恐,id為:"+ id+ ",進(jìn)行裝載到公共緩沖區(qū)中孩哑。。翠桦。");
                if(!this.queue.offer(data,2,TimeUnit.SECONDS)){
                    System.out.print("提交緩沖區(qū)數(shù)據(jù)失敗");
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.print("aaa");
        }
    }

    public void  stop(){
        this.isRunning = false;
    }
}

這里有幾個(gè)注意點(diǎn)横蜒,一個(gè)就是對共享緩沖區(qū)的選擇,作為生產(chǎn)者–消費(fèi)者模型而言,共享緩沖區(qū)一定要具備阻塞的能力丛晌。所以這邊選擇的是阻塞隊(duì)列鹰霍。還有一個(gè)就是在并發(fā)編程的時(shí)候,如果需要使用類似i++這種id自增長的功能茵乱,需要使用Atomic包下的并發(fā)類茂洒。因?yàn)檫@些類是采用CAS設(shè)計(jì)的,不會(huì)產(chǎn)生并發(fā)問題瓶竭。

3.消費(fèi)者

public class Consumer implements Runnable {

    private BlockingQueue<Data> queue;

    public Consumer(BlockingQueue queu){
        this.queue = queu;
    }

    //隨機(jī)對象
    private static Random r = new Random();

    @Override
    public void run() {
        while(true){
            try{
                //獲取數(shù)據(jù)
                Data data = this.queue.take();
                //進(jìn)行數(shù)據(jù)處理督勺,休眠 0-1000毫秒模擬耗時(shí)
                Thread.sleep(r.nextInt(1000));
                System.out.print("當(dāng)前消費(fèi)線程"+Thread.currentThread().getName() +",消費(fèi)成功,消費(fèi)id為"+data.getId());
            }catch(InterruptedException e){
                e.printStackTrace();
            }
        }
    }
}

消費(fèi)者主要就是從阻塞隊(duì)列中獲取數(shù)據(jù)智哀,如果隊(duì)列中沒有元素,則會(huì)釋放CPU荧恍,然后等待。(注意這里使用的是take而不是poll送巡,不同點(diǎn)在于take在沒有元素的時(shí)候會(huì)釋放CPU,而poll則是直接返回null)骗爆。

main函數(shù):

public class Main {
    public static void main(String[] args){
        //內(nèi)存緩沖區(qū)
        BlockingQueue<Data> queue = new LinkedBlockingQueue<Data>(10);
        //生產(chǎn)者
        Provider p1 = new Provider(queue);
        Provider p2 = new Provider(queue);
        Provider p3 = new Provider(queue);

        Consumer c1 = new Consumer(queue);
        Consumer c2 = new Consumer(queue);
        Consumer c3 = new Consumer(queue);

        //創(chuàng)建線程池次氨,這是一個(gè)緩存的線程池摘投,可以創(chuàng)建無窮大的線程,沒有任務(wù)的時(shí)候不創(chuàng)建線程犀呼,空閑線程存活的時(shí)間為60s。
        ExecutorService cachepool = Executors.newCachedThreadPool();
        cachepool.execute(p1);
        cachepool.execute(p2);
        cachepool.execute(p3);
        cachepool.execute(c1);
        cachepool.execute(c2);
        cachepool.execute(c3);
        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        p1.stop();
        p2.stop();
        p3.stop();
        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末外臂,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子专钉,更是在濱河造成了極大的恐慌挑童,老刑警劉巖跃须,帶你破解...
    沈念sama閱讀 211,123評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異娃兽,居然都是意外死亡菇民,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,031評論 2 384
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來第练,“玉大人阔馋,你說我怎么就攤上這事〗刻停” “怎么了呕寝?”我有些...
    開封第一講書人閱讀 156,723評論 0 345
  • 文/不壞的土叔 我叫張陵,是天一觀的道長婴梧。 經(jīng)常有香客問我下梢,道長,這世上最難降的妖魔是什么塞蹭? 我笑而不...
    開封第一講書人閱讀 56,357評論 1 283
  • 正文 為了忘掉前任孽江,我火速辦了婚禮,結(jié)果婚禮上番电,老公的妹妹穿的比我還像新娘岗屏。我一直安慰自己,他們只是感情好漱办,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,412評論 5 384
  • 文/花漫 我一把揭開白布这刷。 她就那樣靜靜地躺著,像睡著了一般娩井。 火紅的嫁衣襯著肌膚如雪崭歧。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,760評論 1 289
  • 那天撞牢,我揣著相機(jī)與錄音率碾,去河邊找鬼。 笑死屋彪,一個(gè)胖子當(dāng)著我的面吹牛所宰,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播畜挥,決...
    沈念sama閱讀 38,904評論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼仔粥,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了蟹但?” 一聲冷哼從身側(cè)響起躯泰,我...
    開封第一講書人閱讀 37,672評論 0 266
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎华糖,沒想到半個(gè)月后麦向,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,118評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡客叉,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,456評論 2 325
  • 正文 我和宋清朗相戀三年诵竭,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了话告。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,599評論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡卵慰,死狀恐怖沙郭,靈堂內(nèi)的尸體忽然破棺而出裳朋,到底是詐尸還是另有隱情,我是刑警寧澤鲤嫡,帶...
    沈念sama閱讀 34,264評論 4 328
  • 正文 年R本政府宣布泛范,位于F島的核電站,受9級特大地震影響罢荡,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜惭缰,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,857評論 3 312
  • 文/蒙蒙 一笼才、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧骡送,春花似錦、人聲如沸虐先。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,731評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽篮愉。三九已至,卻和暖如春猪勇,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背冗酿。 一陣腳步聲響...
    開封第一講書人閱讀 31,956評論 1 264
  • 我被黑心中介騙來泰國打工络断, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留项玛,地道東北人弱判。 一個(gè)月前我還...
    沈念sama閱讀 46,286評論 2 360
  • 正文 我出身青樓昌腰,卻偏偏與公主長得像,于是被迫代替她去往敵國和親遭商。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,465評論 2 348

推薦閱讀更多精彩內(nèi)容