亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

C #中的幾個線程同步對象方法

2019-11-17 04:59:49
字體:
來源:轉載
供稿:網友
在編寫多線程程序時無可避免會碰到線程的同步問題。什么是線程的同步呢?

舉個例子:假如在一個公司里面有一個變量記錄某人T的工資count=100,有兩個主管A和B(即工作線程)在早一些時候拿了這個變量的值回去
,過了一段時間A主管將T的工資加了5塊,并存回count變量,而B主管將T的工資減去3塊,并存回count變量。好了,本來T君可以得到102塊的工資的,現在就變成98塊了。這就是線程同步要解決的問題。

在.Net的某些對象里面,在讀取里面的數據的同時還可以修改數據,這類的對象就是“線程安全”。但對于自己編寫的代碼段而言,就必須使用線程同步技術來保證數據的完整性和正確性了。

有幾個規律:
1、假如一個對象(或變量)不會同時被多個其他線程訪問,那么這個對象是不需使用線程同步的。
2、假如雖然有多個線程同時訪問一個對象,但他們所訪問的數據或方法并不相同(不交叉),那這種情況也不需使用線程同步。
例如上例中的那個公司里面假如有 T 和 Q 兩個人,但他們的工資分別是由 A 和 B 主管的,那么這個工資的處理就不需要線程同步了。
3、假如一個對象會同時被多個其他線程訪問,一般只需為這個對象添加線程同步的代碼,而其他線程是不需添加額外代碼的。

在C#里面用于實現線程同步的常用類有如下幾類
1、Mutex類(互斥器),Monitor類,lock方法
2、ManualResetEvent類,AutoResetEvent類(這兩個都是由EventWaitHandle類派生出來的)
3、ReaderWriterLock類

同一類的作用都差不多:其中
第一類的作用是:用來保護某段代碼在執行的時候以獨占的方式執行,這時假如有第二個線程想訪問這個對象時就會被暫停。一直等到獨占的
代碼執行為止。就好比一堆人同時上一個公共廁所一樣,使用這個方法就可以解決文章一開始時提出的問題:主管A要處理T君的工資之前,先lock一下T君,然后取出目前的count值,處理完之后再解除T君的鎖定。假如主管B在主管A處理工資時也想取出count值,那么它只能是一直地等待A處理完之后才能繼續。使用這個方法的一個缺點就是會降低程序的效率。本來是一個多個線程的操作,一旦碰到lock的語句時,那么這些線程只要排隊處理,形同一個單線程操作。

下面舉個例子說明一下這三個方法的使用:
假定有一個Tools類,里面一個int變量,還有Add和Delete方法,其中Add方法會使int變量的值增加,Delete方法使int變量值減少:

public class Tools
{
PRivate int count = 100;
public void Add(int n)
{
count+=n;
}

public void Delete(int n)
{
count-=n;
}
}

在多個線程同時訪問這段代碼時,因為一個語句會被編譯器編譯成多個指令,所以會可能出現這種情況:但某個線程調用Add方法時,這時的count值為 100,而正當要加上n的時候,另外一個線程調用了Delete,它要減去m,結果count加上了n,然后又在原先count=100的值的情況
下減掉了m,最后的結果是count被減去了m,而沒有加上n。很明顯Add方法和Delete方法是不能同時被調用的,所以必須進行線程同步處理。簡單的方法是用lock語句:

public class Tools
{
private object abcde = new object();
private int count = 100;

public void Add(int n)
{
lock(abcde)
{
count+=n;
}
}

public void Delete(int n)
{
lock(abcde)
{
count-=n;
}
}
}

其中abcde是一個private級的內部變量,它不表示任何的意義,只是作為一種“令牌”的角色。
當執行Add方法中的lock(abcde)方法時,這個令牌就在Add方法的手中了,假如這時有第二個線程也想拿這個令牌,沒門,惟有等待。一旦第一
個lock語句的花括號范圍結束之后,這時令牌就被釋放了,同時會迅速落到第二個線程的手中,并且排除其他后來的人。

使用Monitor類的方法大致一樣:

public class Tools
{
private object abcde = new object();
private int count = 100;

public void Add(int n)
{
Monitor.Enter(abcde);
count+=n;
Monitor.Exit(abcde);
}

public void Delete(int n)
{
Monitor.Enter(abcde);
count-=n;
Monitor.Exit(abcde);
}
}

Monitor的常用方法:Enter和Exit都是靜態方法,作用跟lock語句的兩個花括號一樣。
而使用 Mutex 就不需聲明一個“令牌”對象了,但要實例化之后才可以使用:

public class Tools
{
private Mutex mut = new Mutex();
private int count = 100;

public void Add(int n)
{
mut.WaitOne();
count+=n;
mut.ReleaseMutex();
}

public void Delete(int n)
{
mut.WaitOne();
count-=n;
mut.ReleaseMutex();
}
}

其中的WaitOne為等待方法,一直等到Mutex 被釋放為止。初始的情況下,Mutex 對象是處于釋放狀態的,而一旦執行了WaitOne方法之后,它
就被捕捉了,一直到被調用了ReleaseMutex方法之后才被釋放。

使用這三種方法都有一個要注重的問題,就是在獨占代碼段里面假如引起了異常,可能會使“令牌”對象不被釋放,這樣程序就會一直地死等下去了。
所以要在獨占代碼段里面處理好異常。例如下面這樣的代碼就是錯誤的:

public void Add(int n)
{
try
{
mut.WaitOne();
count+=n;
//....這里省略了N行代碼
//....這里是有可能引起異常的代碼
//....這里省略了N行代碼
mut.ReleaseMutex();
}
catch
{
Console.Writeline("error.");
}
}

上面的代碼一旦在try和catch里面發生了異常,那么Mutex就不能被釋放,后面的程序就會卡死在WaitOne()一行,而應該改成這樣:

public void Add(int n)
{
mut.WaitOne();
try
{
count+=n;
//....這里省略了N行代碼
//....這里是有可能引起異常的代碼
//....這里省略了N行代碼
}
catch
{
Console.Writeline("error.");
}
mut.ReleaseMutex();
}

現在談一下第二種:
ManualResetEvent類,AutoResetEvent類

上面這兩個類都是由EventWaitHandle類派生出來的,所以功能和調用方法都很相似。
這兩個類常用于阻斷某個線程的執行,然后在符合條件的情況下再恢復其執行。
舉個例子,你想送花給一個MM,托了一個送花的小伙子送了過去,而你希望當MM收到花之后就立即打個電話過去告訴她。

但問題是你不知道花什么時候才送到MM的手里,打早了打遲了都不好,這時你可以使用ManualResetEvent對象幫忙。當委

托小伙子送花過去的時候,使用ManualResetEvent的WaitOne方法進行等待。當小伙子把花送到MM的手中時,再調用一下

ManualResetEvent的Set方法,你就可以準時地打電話過去了。
另外ManualResetEvent還有一個Reset方法,用來重新阻斷調用者執行的,情況就好比你委托了這個小伙子送花給N個MM,

而又想準時地給這N個MM打電話的情況一樣。

using System;
using System.Threading;

public class TestMain
{
private static ManualResetEvent ent = new ManualResetEvent(false);

public static void Main()
{
Boy sender = new Boy(ent);
Thread th = new Thread(new ThreadStart(sender.SendFlower));
th.Start();

ent.WaitOne(); //等待工作
Console.WriteLine("收到了吧,花是我送嘀:)");
Console.ReadLine();
}

}

public class Boy
{
ManualResetEvent ent;

public Boy(ManualResetEvent e)
{
ent = e;
}

public void SendFlower()
{
Console.WriteLine("正在送花的途中");
for (int i = 0; i < 10; i++)
{
Thread.Sleep(200);
Console.Write("..");
}
Console.WriteLine("/r/n花已經送到MM手中了,boss");

ent.Set(); //通知阻塞程序
}
}

而AutoResetEvent類故名思意,就是在每次Set完之后自動Reset。讓執行程序重新進入阻塞狀態。
即AutoResetEvent.Set() 相當于 ManualResetEvent.Set() 之后又立即 ManualResetEvent.Reset(),
其他的就沒有什么不同的了。
舉個送花給N個MM的例子:

using System;
using System.Threading;

public class TestMain
{
private static AutoResetEvent ent = new AutoResetEvent(false);

public static void Main()
{
Boy sender = new Boy(ent);

for (int i = 0; i < 3; i++)
{
Thread th = new Thread(new ThreadStart(sender.SendFlower));
th.Start();
ent.WaitOne(); //等待工作
Console.WriteLine("收到了吧,花是我送嘀:)/r/n/r/n");
}

Console.ReadLine();
}

}

public class Boy
{
AutoResetEvent ent;

public Boy(AutoResetEvent e)
{
ent = e;
}

public void SendFlower()
{
Console.WriteLine("正在送花的途中");
for (int i = 0; i < 10; i++)
{
Thread.Sleep(200);
Console.Write("..");
}
Console.WriteLine("/r/n花已經送到MM手中了,boss");

ent.Set(); //通知阻塞程序,這里的效果相當于 ManualResetEvent的Set()方法+Reset()方法

}
}

要注重的是ManualResetEvent和AutoResetEvent 的構造函數都有一個bool的參數,用這個參數可以指定初始情況下,同步對象的處于阻塞(設置為false)還是非阻塞(設置為true)的狀態。
另外WaitOne方法也可以帶兩個參數:
WaitOne (int millisecondsTimeout,bool exitContext)
millisecondsTimeout:等待的毫秒數,或為 Timeout.Infinite (-1),表示無限期等待。
exitContext:為 true,則等待之前先退出上下文的同步域(假如在同步上下文中),然后在稍后重新獲取它;否則為false。
就是說,等待是可以加上一個期限的,假如等待的同步對象一直都不Set()的話,那么程序就會卡死,所以在WaitOne方法里面可以放置一個時間期限,單位是毫秒。


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产区亚洲区欧美区| 美日韩精品免费观看视频| 日韩精品视频在线观看免费| 日韩专区中文字幕| 日韩经典一区二区三区| 91精品久久久久久久久久| 狠狠躁夜夜躁人人爽天天天天97| 国产深夜精品福利| 国产精品一区二区三区免费视频| 成人免费视频网址| 动漫精品一区二区| 亚洲区中文字幕| 91精品在线国产| 日韩久久免费视频| 国产精品青青在线观看爽香蕉| 亚洲国产精久久久久久| 91在线高清免费观看| 日韩视频免费中文字幕| 国产精品自产拍在线观| 91在线免费看网站| 欧美一区深夜视频| 亚洲视频欧洲视频| 亚洲欧美中文日韩在线v日本| 日韩大陆欧美高清视频区| 欧美人与性动交a欧美精品| 国产精品99久久久久久www| 色悠悠久久久久| 91av在线免费观看视频| 欧美巨乳在线观看| 欧亚精品在线观看| 中文字幕免费精品一区| 国产欧美精品久久久| 国产精品激情自拍| 亚洲精品v天堂中文字幕| 亚洲欧美一区二区激情| 欧美日韩激情视频| 亚洲国产成人精品久久久国产成人一区| 91精品国产777在线观看| 欧美亚洲午夜视频在线观看| 欧美在线亚洲在线| 亚洲自拍偷拍区| 欧美激情视频在线免费观看 欧美视频免费一| 91精品啪在线观看麻豆免费| 国产在线精品播放| 日韩精品免费电影| 国产99在线|中文| 激情久久av一区av二区av三区| 日韩在线观看免费高清完整版| 欧美夫妻性视频| 日本91av在线播放| 成人97在线观看视频| 国产日韩中文字幕在线| 91精品视频免费观看| 久久av中文字幕| 国产精品网红直播| 亚洲sss综合天堂久久| 国产精品久久久| 国产精品三级美女白浆呻吟| 视频直播国产精品| 午夜精品国产精品大乳美女| 久久久久北条麻妃免费看| 国产97免费视| 日韩在线观看免费全集电视剧网站| 国产美女精彩久久| 亚洲国产精彩中文乱码av在线播放| 欧美最猛性xxxxx(亚洲精品)| 国产在线高清精品| 2020欧美日韩在线视频| 国产成人精品在线视频| 久久精品国产欧美亚洲人人爽| 亚洲黄页网在线观看| 精品在线欧美视频| 久久久久久久久久婷婷| 中文字幕一区日韩电影| 日韩麻豆第一页| 一区二区三区无码高清视频| 国产一区二区三区久久精品| 日韩欧美一区二区三区| 欧美激情第三页| 8050国产精品久久久久久| 最好看的2019年中文视频| 91精品久久久久| 中文欧美日本在线资源| 日韩美女在线看| 亚洲欧美一区二区三区四区| 日本成熟性欧美| 亚洲xxxx3d| 成人黄色短视频在线观看| 国内揄拍国内精品少妇国语| 亚洲国产精品久久久久秋霞不卡| 国产91对白在线播放| 日本精品视频在线| 国产精品91久久| 中文欧美日本在线资源| 欧美日韩午夜剧场| 亚洲最大福利视频| 亚洲一区二区中文字幕| 91社区国产高清| 日韩中文在线中文网三级| 亚洲毛片在线看| 欧美寡妇偷汉性猛交| 精品调教chinesegay| 日韩美女主播视频| 欧美二区乱c黑人| 亚洲色图激情小说| 一个色综合导航| 亚洲男人天堂2019| 51精品国产黑色丝袜高跟鞋| 91精品国产综合久久久久久久久| 欧美极品xxxx| 最近2019中文字幕mv免费看| 亚洲理论电影网| 97在线看福利| 日韩黄色在线免费观看| 国产精品高潮视频| 国产精品久久久久福利| 永久555www成人免费| 高清日韩电视剧大全免费播放在线观看| 久久久国产精彩视频美女艺术照福利| 在线观看久久av| 91久久嫩草影院一区二区| 日本一区二三区好的精华液| 亚洲free性xxxx护士hd| 日韩中文字幕免费看| 国外色69视频在线观看| 中文字幕久热精品视频在线| 人人澡人人澡人人看欧美| 成人看片人aa| 欧美三级欧美成人高清www| 91在线色戒在线| 茄子视频成人在线| 91久久国产综合久久91精品网站| 成人网页在线免费观看| 欧美国产日韩一区| 久久久久久亚洲精品不卡| 欧美一级淫片aaaaaaa视频| 久久福利网址导航| 亚洲欧美日韩国产中文| 91伊人影院在线播放| 92看片淫黄大片看国产片| 精品国产乱码久久久久久虫虫漫画| 欧美巨猛xxxx猛交黑人97人| 国产日产久久高清欧美一区| 亚洲**2019国产| 国产又爽又黄的激情精品视频| 日韩精品亚洲精品| 欧美中文字幕第一页| 国产成人+综合亚洲+天堂| 欧美最顶级的aⅴ艳星| 国产精品对白刺激| 美女扒开尿口让男人操亚洲视频网站| 欧洲美女7788成人免费视频| 亚洲成人中文字幕| 97香蕉超级碰碰久久免费软件| 午夜精品免费视频| 色悠悠国产精品| 欧美久久精品午夜青青大伊人| 51久久精品夜色国产麻豆| 久久亚洲综合国产精品99麻豆精品福利| 亚洲黄色在线观看| 亚洲色无码播放| 久久久久久久久久久国产| 在线观看国产精品淫| 黑人巨大精品欧美一区二区三区|