日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > C# >内容正文

C#

[转]C#多线程学习(三) 生产者和消费者

發布時間:2025/1/21 C# 78 豆豆
生活随笔 收集整理的這篇文章主要介紹了 [转]C#多线程学习(三) 生产者和消费者 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

前面說過,每個線程都有自己的資源,但是代碼區是共享的,即每個線程都可以執行相同的函數。這可能帶來的問題就是幾個線程同時執行一個函數,導致數據的混亂,產生不可預料的結果,因此我們必須避免這種情況的發生。


C#提供了一個關鍵字lock,它可以把一段代碼定義為互斥段(critical section),互斥段在一個時刻內只允許一個線程進入執行,而其他線程必須等待。在C#中,關鍵字lock定義如下:

lock(expression) statement_block?
?
expression代表你希望跟蹤的對象,通常是對象引用。
??? 如果你想保護一個類的實例,一般地,你可以使用this;
??? 如果你想保護一個靜態變量(如互斥代碼段在一個靜態方法內部),一般使用類名就可以了。

而statement_block就是互斥段的代碼,這段代碼在一個時刻內只可能被一個線程執行。


下面是一個使用lock關鍵字的典型例子,在注釋里說明了lock關鍵字的用法和用途。
示例如下:

usingSystem;
usingSystem.Threading;

namespaceThreadSimple
{
????internalclassAccount?
????{
????????intbalance;
????????Random?r?=newRandom();
????????
????????internalAccount(intinitial)?
????????{
????????????balance?=initial;
????????}?

????????internalintWithdraw(intamount)?
????????{
????????????if(balance?<0)
????????????{
????????????????//如果balance小于0則拋出異常
thrownewException("Negative?Balance");
????????????}
????????????//下面的代碼保證在當前線程修改balance的值完成之前
????????????//不會有其他線程也執行這段代碼來修改balance的值
????????????//因此,balance的值是不可能小于0?的
lock(this)
????????????{
????????????????Console.WriteLine("Current?Thread:"+Thread.CurrentThread.Name);
????????????????//如果沒有lock關鍵字的保護,那么可能在執行完if的條件判斷之后
????????????????//另外一個線程卻執行了balance=balance-amount修改了balance的值
????????????????//而這個修改對這個線程是不可見的,所以可能導致這時if的條件已經不成立了
????????????????//但是,這個線程卻繼續執行balance=balance-amount,所以導致balance可能小于0
if(balance?>=amount)?
????????????????{
????????????????????Thread.Sleep(5);
????????????????????balance?=balance?-amount;
????????????????????returnamount;
????????????????}?
????????????????else
????????????????{
????????????????????return0;?//?transaction?rejected
}
????????????}
????????}
????????internalvoidDoTransactions()?
????????{
????????????for(inti?=0;?i?<100;?i++)?
????????????Withdraw(r.Next(-50,?100));
????????}
????}?

????internalclassTest?
????{
????????staticinternalThread[]?threads?=newThread[10];
????????publicstaticvoidMain()?
????????{
????????????Account?acc?=newAccount?(0);
????????????for(inti?=0;?i?<10;?i++)?
????????????{
????????????????Thread?t?=newThread(newThreadStart(acc.DoTransactions));
????????????????threads[i]?=t;
????????????}
????????????for(inti?=0;?i?<10;?i++)?
????????????????threads[i].Name=i.ToString();
????????????for(inti?=0;?i?<10;?i++)?
????????????????threads[i].Start();
????????????Console.ReadLine();
????????}
????}
}

?


Monitor 類鎖定一個對象

當多線程公用一個對象時,也會出現和公用代碼類似的問題,這種問題就不應該使用lock關鍵字了,這里需要用到System.Threading中的一個類Monitor,我們可以稱之為監視器,Monitor提供了使線程共享資源的方案。

  Monitor類可以鎖定一個對象,一個線程只有得到這把鎖才可以對該對象進行操作。對象鎖機制保證了在可能引起混亂的情況下一個時刻只有一個線程可以訪問這個對象。
Monitor必須和一個具體的對象相關聯,但是由于它是一個靜態的類,所以不能使用它來定義對象,而且它的所有方法都是靜態的,不能使用對象來引用。下面代碼說明了使用Monitor鎖定一個對象的情形:

......
Queue oQueue=new Queue();
......
Monitor.Enter(oQueue);
......//現在oQueue對象只能被當前線程操縱了
Monitor.Exit(oQueue);//釋放鎖?
?

如上所示,當一個線程調用Monitor.Enter()方法鎖定一個對象時,這個對象就歸它所有了,其它線程想要訪問這個對象,只有等待它使用Monitor.Exit()方法釋放鎖。為了保證線程最終都能釋放鎖,你可以把Monitor.Exit()方法寫在try-catch-finally結構中的finally代碼塊里。

對于任何一個被Monitor鎖定的對象,內存中都保存著與它相關的一些信息:
其一是現在持有鎖的線程的引用;
其二是一個預備隊列,隊列中保存了已經準備好獲取鎖的線程;
其三是一個等待隊列,隊列中保存著當前正在等待這個對象狀態改變的隊列的引用。

當擁有對象鎖的線程準備釋放鎖時,它使用Monitor.Pulse()方法通知等待隊列中的第一個線程,于是該線程被轉移到預備隊列中,當對象鎖被釋放時,在預備隊列中的線程可以立即獲得對象鎖。


下面是一個展示如何使用lock關鍵字和Monitor類來實現線程的同步和通訊的例子,也是一個典型的生產者與消費者問題。
這個例程中,生產者線程和消費者線程是交替進行的,生產者寫入一個數,消費者立即讀取并且顯示(注釋中介紹了該程序的精要所在)。

用到的系統命名空間如下:
using System;
using System.Threading;

首先,定義一個被操作的對象的類Cell,在這個類里,有兩個方法:ReadFromCell()和WriteToCell。消費者線程將調用ReadFromCell()讀取cellContents的內容并且顯示出來,生產者進程將調用WriteToCell()方法向cellContents寫入數據。

示例如下:

public?class?Cell
{
????????int?cellContents;?//Cell對象里邊的內容
????????bool?readerFlag?=?false;?//狀態標志,為true時可以讀取,為false則正在寫入
????????public?int?ReadFromCell(?)
????????{
????????????lock(this)?//Lock關鍵字保證了什么,請大家看前面對lock的介紹
????????????{
????????????????if?(!readerFlag)//如果現在不可讀取
????????????????{?
????????????????????try
????????????????????{
????????????????????????//等待WriteToCell方法中調用Monitor.Pulse()方法
????????????????????????Monitor.Wait(this);
????????????????????}
????????????????????catch?(SynchronizationLockException?e)
????????????????????{
????????????????????????Console.WriteLine(e);
????????????????????}
????????????????????catch?(ThreadInterruptedException?e)
????????????????????{
????????????????????????Console.WriteLine(e);
????????????????????}
????????????????}
????????????????Console.WriteLine("Consume:?{0}",cellContents);
????????????????readerFlag?=?false;
????????????????//重置readerFlag標志,表示消費行為已經完成
????????????????Monitor.Pulse(this);?
????????????????//通知WriteToCell()方法(該方法在另外一個線程中執行,等待中)
????????????}
????????????return?cellContents;
????????}
????
????????public?void?WriteToCell(int?n)
????????{
????????????lock(this)
????????????{
????????????????if?(readerFlag)
????????????????{
????????????????????try
????????????????????{
????????????????????????Monitor.Wait(this);
????????????????????}
????????????????????catch?(SynchronizationLockException?e)
????????????????????{
????????????????????????????//當同步方法(指Monitor類除Enter之外的方法)在非同步的代碼區被調用
????????????????????????Console.WriteLine(e);
????????????????????}
????????????????????catch?(ThreadInterruptedException?e)
????????????????????{
????????????????????????????//當線程在等待狀態的時候中止?
????????????????????????Console.WriteLine(e);
????????????????????}
????????????????}
????????????????cellContents?=?n;
????????????????Console.WriteLine("Produce:?{0}",cellContents);
????????????????readerFlag?=?true;?
????????????????Monitor.Pulse(this);?
????????????????//通知另外一個線程中正在等待的ReadFromCell()方法
????????????}
????????}
}


下面定義生產者類 CellProd 和消費者類 CellCons ,它們都只有一個方法ThreadRun(),以便在Main()函數中提供給線程的ThreadStart代理對象,作為線程的入口。

public?class?CellProd
{
  ????Cell?cell;?//被操作的Cell對象
  ????int?quantity?=?1;?//生產者生產次數,初始化為1?

  ????public?CellProd(Cell?box,?int?request)
  ????{
????????//構造函數
????????cell?=?box;?
????????quantity?=?request;?
  ????}
  ????public?void?ThreadRun(?)
  ????{
????????for(int?looper=1;?looper<=quantity;?looper++)
    ????????cell.WriteToCell(looper);?//生產者向操作對象寫入信息
  ????}
}

public?class?CellCons
{
  ????Cell?cell;?
  ????int?quantity?=?1;?

  ????public?CellCons(Cell?box,?int?request)
  ????{
????????????????//構造函數
????????cell?=?box;?
????????quantity?=?request;?
  ????}
  ????public?void?ThreadRun(?)
  ????{
????????int?valReturned;
????????for(int?looper=1;?looper<=quantity;?looper++)
    ????????valReturned=cell.ReadFromCell(?);//消費者從操作對象中讀取信息
  ????}
}?


然后在下面這個類MonitorSample的Main()函數中,我們要做的就是創建兩個線程分別作為生產者和消費者,使用CellProd.ThreadRun()方法和CellCons.ThreadRun()方法對同一個Cell對象進行操作。

public?class?MonitorSample
{
  ????public?static?void?Main(String[]?args)
  ????{
????????int?result?=?0;?//一個標志位,如果是0表示程序沒有出錯,如果是1表明有錯誤發生
????????Cell?cell?=?new?Cell(?);?

????????//下面使用cell初始化CellProd和CellCons兩個類,生產和消費次數均為20次
????????CellProd?prod?=?new?CellProd(cell,?20);?
????????CellCons?cons?=?new?CellCons(cell,?20);?

????????Thread?producer?=?new?Thread(new?ThreadStart(prod.ThreadRun));
????????Thread?consumer?=?new?Thread(new?ThreadStart(cons.ThreadRun));
????????//生產者線程和消費者線程都已經被創建,但是沒有開始執行?
????????try
????????{
    ????producer.Start(?);
    ????consumer.Start(?);?

    ????producer.Join(?);?
    ????consumer.Join(?);
    ????Console.ReadLine();
????????}
????????catch?(ThreadStateException?e)
????????{
    ????//當線程因為所處狀態的原因而不能執行被請求的操作
    ????Console.WriteLine(e);?
    ????result?=?1;?
????????}
????????catch?(ThreadInterruptedException?e)
????????{
    ????//當線程在等待狀態的時候中止
    ????Console.WriteLine(e);?
    ????result?=?1;?
????????}
????????//盡管Main()函數沒有返回值,但下面這條語句可以向父進程返回執行結果
????????Environment.ExitCode?=?result;
  ????}
}


在上面的例程中,同步是通過等待Monitor.Pulse()來完成的。首先生產者生產了一個值,而同一時刻消費者處于等待狀態,直到收到生產者的“脈沖(Pulse)”通知它生產已經完成,此后消費者進入消費狀態,而生產者開始等待消費者完成操作后將調用Monitor.Pulese()發出的“脈沖”。

它的執行結果很簡單:

Produce: 1
Consume: 1
Produce: 2
Consume: 2
Produce: 3
Consume: 3
...
...
Produce: 20
Consume: 20?
?
事實上,這個簡單的例子已經幫助我們解決了多線程應用程序中可能出現的大問題,只要領悟了解決線程間沖突的基本方法,很容易把它應用到比較復雜的程序中去。

文章來源:http://www.cnblogs.com/xugang/archive/2008/03/23/1118594.html

轉載于:https://www.cnblogs.com/yuanyl/p/3234312.html

總結

以上是生活随笔為你收集整理的[转]C#多线程学习(三) 生产者和消费者的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

主站蜘蛛池模板: 朝桐光av在线 | 最新黄网| 婷婷久久久久久 | 久久久久激情 | 久久久久亚洲av成人片 | 欧洲国产视频 | 日韩高清在线播放 | 看成人片 | 福利姬在线观看 | 亚洲熟女乱色一区二区三区 | 污污的网站在线免费观看 | 国产精品久久久久久久天堂 | 亚洲精品国产精品国自产观看 | 久久天天躁狠狠躁夜夜躁2014 | 日本三级视频 | 日韩激情电影在线 | 成人性生生活性生交全黄 | 床戏高潮做进去大尺度视频网站 | 九九视频免费看 | 日韩精品一区二区三区在线观看 | 午夜精品久久久久 | 国产精品永久免费视频 | 91污网站| 91国产视频在线观看 | 日韩私人影院 | 中文字幕日本人妻久久久免费 | 国产精品成人自拍 | 懂色av,蜜臀av粉嫩av | 91麻豆国产在线 | 夜夜摸夜夜爽 | 最近最新中文字幕 | 亚洲乱色熟女一区二区三区 | 亚洲一区二区三区国产 | 男女日批 | 免费国产 | 免费a在线观看播放 | 特大黑人巨人吊xxxx | 亚洲少妇网站 | 欧美人与性动交α欧美精品 | 成人网免费看 | 中文字幕一区二区在线老色批影视 | 欧美日韩视频一区二区三区 | 奇米影视第4色 | 三度诱惑免费版电影在线观看 | 97精品一区 | 50度灰在线 | 日本熟妇色xxxxx日本免费看 | a网站在线 | 欧美a级片在线观看 | 成人吃奶视频 | 国产成人影视 | 黄色片xxxx| 欧美成人精品在线观看 | 每日更新av | 中文字幕人妻丝袜二区 | 精品久草 | 一本在线 | 亚洲 欧美 日韩 综合 | 91精品999| 日韩高清三区 | 国产女人高潮视频 | 黄色国产大片 | 欧美日韩国产片 | 中文字幕乱码一区二区 | 极品少妇xxxx精品少妇 | 中出一区| www.色com | 天天操天天操天天 | 久久久久毛片 | 熟女人妻aⅴ一区二区三区60路 | 国内久久精品视频 | 男女洗澡互摸私密部位视频 | 国产成人精品电影 | 男女又爽又黄 | 精品99久久 | 亚洲色图第1页 | 爱情岛论坛亚洲品质自拍 | 小早川怜子一区二区三区 | 日本二三区| 真实的国产乱xxxx在线 | 色综合久久网 | 最新成人| 日韩干| 91porn破解版 | 一区二区av | a毛片毛片av永久免费 | 性高潮在线观看 | 亚洲永久免费视频 | 性生活免费网站 | 亚洲成人精品久久久 | 少妇又色又紧又黄又刺激免费 | 玩弄少妇人妻 | 三年在线观看视频 | 国产视频在线一区 | 欧美123区| 国产伦精品一区二区三区妓女下载 | 中文字幕第二区 | 夜夜操夜夜骑 | 瑟瑟综合网|