網(wǎng)站首頁 編程語言 正文
一、簡(jiǎn)介
線程安全概念:線程安全是指在當(dāng)一個(gè)線程訪問該類的某個(gè)數(shù)據(jù)時(shí),進(jìn)行保護(hù),其他線程不能進(jìn)行訪問直到該線程讀取完,其他線程才可使用。不會(huì)出現(xiàn)數(shù)據(jù)不一致或者數(shù)據(jù)污染。
線程有可能和其他線程共享一些資源,比如,內(nèi)存,文件,數(shù)據(jù)庫等。當(dāng)多個(gè)線程同時(shí)讀寫同一份共享資源的時(shí)候,可能會(huì)引起沖突。這時(shí)候,我們需要引入線程“同步”機(jī)制,即各位線程之間要有個(gè)先來后到,不能一窩蜂擠上去搶作一團(tuán)。線程同步的真實(shí)意思和字面意思恰好相反。線程同步的真實(shí)意思,其實(shí)是“排隊(duì)”:幾個(gè)線程之間要排隊(duì),一個(gè)一個(gè)對(duì)共享資源進(jìn)行操作,而不是同時(shí)進(jìn)行操作。
二、代碼
下面將通過簡(jiǎn)單的四個(gè)案例進(jìn)行對(duì)比,來講解LOCK的實(shí)現(xiàn)線程同步使用。
案例一:
首先創(chuàng)建兩個(gè)線程,兩個(gè)線程執(zhí)行同一個(gè)方法,代碼如下:
class Program
{
static void Main(string[] args)
{
Thread threadA = new Thread(ThreadMethod); //執(zhí)行的必須是無返回值的方法
threadA.Name = "threadA";
Thread threadB = new Thread(ThreadMethod); //執(zhí)行的必須是無返回值的方法
threadB.Name = "threadB";
threadA.Start();
threadB.Start();
Console.ReadKey();
}
public static void ThreadMethod(object parameter)
{
for (int i = 1; i <= 10; i++)
{
Console.WriteLine("我是:{0},我循環(huán){1}次", Thread.CurrentThread.Name, i);
Thread.Sleep(1000);//休眠一秒
}
}
}
通過下面的執(zhí)行結(jié)果,可以很清楚的看到,兩個(gè)線程是在同時(shí)執(zhí)行ThreadMethod這個(gè)方法,這顯然不符合我們線程同步的要求。
執(zhí)行結(jié)果:
案例二:
通過對(duì)上面代碼的修改如下:
class Program
{
static void Main(string[] args)
{
Program pro = new Program();
Thread threadA = new Thread(pro.ThreadMethod);
threadA.Name = "threadA";
Thread threadB = new Thread(pro.ThreadMethod);
threadB.Name = "threadB";
threadA.Start();
threadB.Start();
Console.ReadKey();
}
public void ThreadMethod(object parameter)
{
lock (this) //添加lock關(guān)鍵字
{
for (int i = 1; i <= 10; i++)
{
Console.WriteLine("我是:{0},我循環(huán){1}次", Thread.CurrentThread.Name, i);
Thread.Sleep(1000);//休眠一秒
}
}
}
}
執(zhí)行結(jié)果:
?
我們通過添加了 lock(this) {...}代碼,查看執(zhí)行結(jié)果實(shí)現(xiàn)了我們想要的線程同步需求。
案例三:
但是我們知道this表示當(dāng)前類實(shí)例的本身,那么有這么一種情況,我們把需要訪問的方法所在的類型進(jìn)行兩個(gè)實(shí)例A和B,線程A訪問實(shí)例A的方法ThreadMethod,線程B訪問實(shí)例B的方法ThreadMethod,這樣的話還能夠達(dá)到線程同步的需求嗎?
修改后的代碼如下:
class Program
{
static void Main(string[] args)
{
Program pro1 = new Program();
Program pro2 = new Program();
Thread threadA = new Thread(pro1.ThreadMethod);
threadA.Name = "threadA";
Thread threadB = new Thread(pro2.ThreadMethod);
threadB.Name = "threadB";
threadA.Start();
threadB.Start();
Console.ReadKey();
}
public void ThreadMethod(object parameter)
{
lock (this) //添加lock關(guān)鍵字
{
for (int i = 1; i <= 10; i++)
{
Console.WriteLine("我是:{0},我循環(huán){1}次", Thread.CurrentThread.Name, i);
Thread.Sleep(1000);//休眠一秒
}
}
}
}
執(zhí)行結(jié)果:
我們會(huì)發(fā)現(xiàn),線程又沒有實(shí)現(xiàn)同步了!lock(this)對(duì)于這種情況是不行的!
案例四:
通過對(duì)上面代碼再次進(jìn)行如下修改:
class Program
{
private static object obj = new object();
static void Main(string[] args)
{
Program pro1 = new Program();
Program pro2 = new Program();
Thread threadA = new Thread(pro1.ThreadMethod);
threadA.Name = "threadA";
Thread threadB = new Thread(pro2.ThreadMethod);
threadB.Name = "threadB";
threadA.Start();
threadB.Start();
Console.ReadKey();
}
public void ThreadMethod(object parameter)
{
lock (obj) //添加lock關(guān)鍵字
{
for (int i = 1; i <= 10; i++)
{
Console.WriteLine("我是:{0},我循環(huán){1}次", Thread.CurrentThread.Name, i);
Thread.Sleep(1000);//休眠一秒
}
}
}
}
執(zhí)行結(jié)果:
通過查看執(zhí)行結(jié)果。會(huì)發(fā)現(xiàn)代碼實(shí)現(xiàn)了我們的需求。
那么 lock(this) 和lock(Obj)有什么區(qū)別呢?
lock(this) 鎖定 當(dāng)前實(shí)例對(duì)象,如果有多個(gè)類實(shí)例的話,lock鎖定的只是當(dāng)前類實(shí)例,對(duì)其它類實(shí)例無影響。所有不推薦使用。
lock(typeof(Model))鎖定的是model類的所有實(shí)例。
lock(obj)鎖定的對(duì)象是全局的私有化靜態(tài)變量。外部無法對(duì)該變量進(jìn)行訪問。
lock 確保當(dāng)一個(gè)線程位于代碼的臨界區(qū)時(shí),另一個(gè)線程不進(jìn)入臨界區(qū)。如果其他線程試圖進(jìn)入鎖定的代碼,則它將一直等待(即被阻止),直到該對(duì)象被釋放。
所以,lock的結(jié)果好不好,還是關(guān)鍵看鎖的誰,如果外邊能對(duì)這個(gè)誰進(jìn)行修改,lock就失去了作用。所以一般情況下,使用私有的、靜態(tài)的并且是只讀的對(duì)象
三、總結(jié)
- 1.lock的是必須是引用類型的對(duì)象,string類型除外。
- 2.lock推薦的做法是使用靜態(tài)的、只讀的、私有的對(duì)象。
- 3.保證lock的對(duì)象在外部無法修改才有意義,如果lock的對(duì)象在外部改變了,對(duì)其他線程就會(huì)暢通無阻,失去了lock的意義。
不能鎖定字符串,鎖定字符串尤其危險(xiǎn),因?yàn)樽址还舱Z言運(yùn)行庫 (CLR)“暫留”。 這意味著整個(gè)程序中任何給定字符串都只有一個(gè)實(shí)例,就是這同一個(gè)對(duì)象表示了所有運(yùn)行的應(yīng)用程序域的所有線程中的該文本。因此,只要在應(yīng)用程序進(jìn)程中的任何位置處具有相同內(nèi)容的字符串上放置了鎖,就將鎖定應(yīng)用程序中該字符串的所有實(shí)例。通常,最好避免鎖定 public 類型或鎖定不受應(yīng)用程序控制的對(duì)象實(shí)例。例如,如果該實(shí)例可以被公開訪問,則 lock(this) 可能會(huì)有問題,因?yàn)椴皇芸刂频拇a也可能會(huì)鎖定該對(duì)象。這可能導(dǎo)致死鎖,即兩個(gè)或更多個(gè)線程等待釋放同一對(duì)象。出于同樣的原因,鎖定公共數(shù)據(jù)類型(相比于對(duì)象)也可能導(dǎo)致問題。而且lock(this)只對(duì)當(dāng)前對(duì)象有效,如果多個(gè)對(duì)象之間就達(dá)不到同步的效果。lock(typeof(Class))與鎖定字符串一樣,范圍太廣了。
原文鏈接:https://www.cnblogs.com/wml-it/p/14821998.html
相關(guān)推薦
- 2023-04-22 Python中DataFrame與內(nèi)置數(shù)據(jù)結(jié)構(gòu)相互轉(zhuǎn)換的實(shí)現(xiàn)_python
- 2022-09-19 教你使用Python從文件中提取IP地址_python
- 2022-09-07 解析react?函數(shù)組件輸入卡頓問題?usecallback?react.memo_React
- 2022-07-17 SQL?Server格式轉(zhuǎn)換函數(shù)Cast、Convert介紹_MsSql
- 2021-12-10 Android?Activity活動(dòng)頁面跳轉(zhuǎn)與頁面?zhèn)髦礯Android
- 2024-03-23 asp.net web api 用戶身份驗(yàn)證
- 2022-02-24 Go入門所踩過的坑:cannot find package "" in any of
- 2022-10-17 C++模擬實(shí)現(xiàn)vector的示例代碼_C 語言
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡(jiǎn)單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支