日韩性视频-久久久蜜桃-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#多线程学习(三) 生产者和消费者的全部內容,希望文章能夠幫你解決所遇到的問題。

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

主站蜘蛛池模板: 亚洲第一中文字幕 | 国产伦精品一区二区三区四区免费 | 看全黄大色黄大片 | 伊人69| 日本免费一区二区三区四区五六区 | 国产噜噜噜 | 一级特黄bbbbb免费观看 | 国产伦子伦对白视频 | 欧洲亚洲天堂 | 一区二区三区四区高清视频 | 日韩中文在线字幕 | 欧美 日韩 国产 中文 | 成人av图片 | 特级淫片aaaaaaa级附近的 | 国产精品夜色一区二区三区 | 一区二区传媒有限公司 | 精品在线视频一区二区 | 亚洲乱视频 | 国产小视频在线观看免费 | 九色综合网| 欧美激情一区二区三级高清视频 | 日韩欧美视频网站 | av资源免费观看 | 日韩成人av一区 | 国产精品23p | 亚洲精品久久久久久久久久久久久 | 青青草91视频 | 国产夜夜嗨 | 久久精品丝袜高跟鞋 | 婷婷中文字幕 | 九九色网| 亚洲情在线 | 日批免费看 | 99re久久精品国产 | 亚洲视频一区二区三区在线观看 | 在线视频播放大全 | 欧美a性| 欧美乱妇一区二区三区 | www操操操| 午夜视频导航 | 99九九久久 | 欧美高清一区二区三区四区 | 中文字幕乱码在线观看 | 一级黄色片在线 | 毛片成人网 | 国产美女无遮挡免费视频 | 亚洲嫩草影院 | 精品国产乱码久久久久久久软件 | 国产一区二区三区视频播放 | 成熟妇人a片免费看网站 | 欧美成人免费观看 | 免费看的黄色 | 天堂中文视频在线 | 久久国内精品 | 成人h在线观看 | 欧美国产大片 | 亚洲av永久无码精品放毛片 | wwwsss在线观看| 久久色资源网 | 久久午夜精品人妻一区二区三区 | 色播视频在线观看 | 亚洲xxxx视频| 午夜久久乐 | 天天想你在线观看完整版电影免费 | 欧洲精品码一区二区三区免费看 | 国产亚洲91| 青青草在线播放 | 视频一区二区免费 | 日本乱子伦xxxx | 欧美日批视频 | 亚洲一级精品 | 日韩欧美精品在线观看 | 欧美一区二区三区影视 | 精品一区久久久 | 香蕉依人 | 天天热天天干 | 亚洲日本色图 | 国产伦理吴梦梦伦理 | 五月天av在线 | 国产成人aaa | 亚洲精品欧洲 | 欧美日一区二区三区 | 秋霞午夜伦理 | 日韩少妇一区二区 | 中文字幕在线观看第一页 | 国产真实生活伦对白 | 日韩精品在线观看一区二区 | 日本大尺度吃奶做爰久久久绯色 | 污网站免费观看 | 欧美激情15p | 中文字幕福利视频 | 好紧好爽再浪一点视频 | 欧美九九| 成人观看| 91污在线观看| 性xxxxxxxxx18欧美 | 久久国产精品久久国产精品 | 中国吞精videos露脸 | 色综合色综合色综合 |