欧美性猛交xxxx免费看_牛牛在线视频国产免费_天堂草原电视剧在线观看免费_国产粉嫩高清在线观看_国产欧美日本亚洲精品一5区

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

什么是線程安全?如何理解線程安全?

冬至子 ? 來源:Linux大陸 ? 作者:LinuxZn ? 2023-05-30 14:33 ? 次閱讀

線程安全

在多線程編程中,線程安全是必須要考慮的因素。

什么是線程安全?

在多線程環(huán)境中,多個線程在同一時刻對同一份資源進(jìn)行寫操作時,不會出現(xiàn)數(shù)據(jù)不一致。反之,則是線程非安全的。

線程安全是程序設(shè)計中的術(shù)語,指某個函數(shù)、函數(shù)庫在多線程環(huán)境中被調(diào)用時,能夠正確地處理多個線程之間的公用變量,使程序功能正確完成。

為了確保在多線程環(huán)境中的線程安全,就要確保數(shù)據(jù)的一致性。確保線程安全的幾種方法:

使用互斥鎖

一個線程,如果需要訪問公共資源,需要獲得互斥鎖并對其加鎖,資源在在鎖定過程中,如果其它線程對其進(jìn)行訪問,也需要獲得互斥鎖,如果獲取不到,線程只能進(jìn)行阻塞,直到獲得該鎖的線程解鎖。

#include 

int increment_counter(void)
{
 static int counter = 0;
 static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;

 pthread_mutex_lock(&mutex);
 
 // only allow one thread to increment at a time
 ++counter;
 // store value before any other threads increment it further
 int result = counter; 

 pthread_mutex_unlock(&mutex);
 
 return result;
}

這個函數(shù)是線程安全的,可以在多個線程中被調(diào)用。

使用原子操作

上面的例子中,使用一個 互斥鎖來保護(hù)一次簡單的增量操作顯然過于昂貴,我們可以使用一些專門的原子操作API函數(shù)來替代。如上述例子,c++11中的原子變量提供了一個可使此函數(shù)既線程安全又可重入(而且還更簡潔)的替代方案:

#include 

int increment_counter(void)
{
 static std::atomic<int> counter(0);
 
 // increment is guaranteed to be done atomically
 int result = ++counter;

 return result;
}

Linux內(nèi)核中原子整形操作:

#include 

int increment_counter(void)
{
 atomic_t counter = ATOMIC_INIT(0);
 
 // increment is guaranteed to be done atomically
 atomic_inc(&counter);
 int result = counter;

 return result;
}

什么是原子操作?

從字面上簡單理解,原子是一種很微小的粒子;原子操作是不能再進(jìn)一步細(xì)分的操作。

從上面互斥鎖的例子來看,在線程層面,線程1和線程2同時調(diào)用了increment_counter函數(shù),被 mutex 保護(hù)的操作是原子操作,lock、unlock及保護(hù)部分要整體順序運(yùn)行,不可再進(jìn)一步細(xì)分,作為一個原子存在 。

如果確定某個操作是原子的,并且有原子操作API函數(shù)可以使用,就不用為了去保護(hù)這個操作而加上會耗費昂貴性能開銷的鎖。

如,Linux內(nèi)核原子整形操作 API 函數(shù)表(來源:正點原子) :

圖片

防止過度優(yōu)化

線程安全的函數(shù)應(yīng)該為每個調(diào)用它的線程分配專門的空間,把多個線程共享的變量正確對待(如,通知編譯器該變量為“易失(volatile)”型,阻止其進(jìn)行一些不恰當(dāng)?shù)膬?yōu)化)。

線程安全函數(shù)與可重入函數(shù)?

先明確概念:

  • 線程安全函數(shù):能夠正確地處理多個線程之間的公用變量的函數(shù)。、
  • 可重入函數(shù):在任意時刻被中斷然后操作系統(tǒng)調(diào)度執(zhí)行另一段代碼,這段代碼又使用了該副程序不會出錯。

可重入函數(shù)應(yīng)當(dāng)滿足條件:

  • 不能含有靜態(tài)(全局)非常量數(shù)據(jù)。
  • 不能返回靜態(tài)(全局)非常量數(shù)據(jù)的地址。
  • 只能處理由調(diào)用者提供的數(shù)據(jù)。
  • 不能依賴于單例模式資源的鎖。
  • 調(diào)用(call)的函數(shù)也必需是可重入的。

可重入函數(shù)未必是線程安全的;線程安全函數(shù)未必是可重入的。

例子1:上述例子中的increment_counter函數(shù)是線程安全的,但是并不是可重入的。因為使用了互斥鎖,如果這個函數(shù)用在可重入的中斷處理程序中,如果在pthread_mutex_lock(&mutex)和pthread_mutex_unlock(&mutex)之間產(chǎn)生另一個調(diào)用函數(shù)increment_counter的中斷,則會第二次執(zhí)行此函數(shù),此時由于mutex已被lock,函數(shù)會在pthread_mutex_lock(&mutex)處阻塞,并且由于mutex沒有機(jī)會被unlock,阻塞會永遠(yuǎn)持續(xù)下去。

例子2:一個函數(shù)打開某個文件并讀入數(shù)據(jù)。這個函數(shù)是可重入的,因為它的多個實例同時執(zhí)行不會造成沖突;但它不是線程安全的,因為在它讀入文件時可能有別的線程正在修改該文件,為了線程安全必須對文件加“同步鎖”。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 編譯器
    +關(guān)注

    關(guān)注

    1

    文章

    1642

    瀏覽量

    49312
  • C++語言
    +關(guān)注

    關(guān)注

    0

    文章

    147

    瀏覽量

    7037
  • LINUX內(nèi)核
    +關(guān)注

    關(guān)注

    1

    文章

    316

    瀏覽量

    21749
收藏 人收藏

    評論

    相關(guān)推薦

    不同創(chuàng)建線程安全Set的方式

    線程安全的問題,真的算是老生常談了。這幾天看到一個 HashSet 線程安全的騷操作,在這里分享給大家。 在本文中,我們將分享如何構(gòu)造線程
    的頭像 發(fā)表于 09-25 14:20 ?720次閱讀

    調(diào)用非安全線程的dll的問題

    在調(diào)用非線程安全的dll時,是不是要選擇在UI線程中運(yùn)行?是不是還必須用不可重入的子VI封裝一下?上述的兩步是不是都要做?這些問題不是很清楚,還請各位大神指點一下
    發(fā)表于 03-14 21:13

    XC32源碼和字符串線程安全

    我正在嘗試我的項目的FrReTOS遷移,我想了解哪些字符串處理函數(shù)是線程安全的,或者不是線程安全的。特別是,我想看看StrudStruts,Strutk,還有一些其他的源代碼。但是,在
    發(fā)表于 11-26 16:03

    YYKit源碼線程安全計數(shù)器YYSentinel

    YYKit源碼探究(八十三) —— 線程安全計數(shù)器YYSentinel(一)
    發(fā)表于 04-28 16:57

    Linux下的線程安全是什么

    Linux下的線程安全原文結(jié)構(gòu)有點亂線程安全:多個執(zhí)行流對臨界資源進(jìn)行爭搶訪問,而不會造成數(shù)據(jù)二義性和邏輯混亂,成這段代碼的過程是線程
    發(fā)表于 07-01 13:34

    什么是線程安全?如何去實現(xiàn)線程安全?

    什么是線程安全?如何去實現(xiàn)線程安全?互斥實現(xiàn)的技術(shù)是什么?有哪些注意事項?同步實現(xiàn)的技術(shù)是什么?其操作流程有哪些?
    發(fā)表于 07-23 09:57

    請教大神rtthread中的ringbuff是線程安全的嗎

    最近想用輕量級的ringbuff,請教大神rtthread中的ringbuff是線程安全的嗎?
    發(fā)表于 07-29 10:44

    什么是線程安全

    線程安全的鏈表-隊列-棧,就是多線程同時操作(包括查找、添加、刪除等)鏈表、隊列或棧,無論如何操作,就是多線程同時操作(包括查找、添加、刪除等)鏈表、隊列或棧,無論如何操作,都不會產(chǎn)生
    發(fā)表于 11-17 11:16 ?1次下載

    解決線程安全問題技巧匯總

    線程,有時被稱為輕量級進(jìn)程,是程序執(zhí)行流的最小單元。一個標(biāo)準(zhǔn)的線程線程ID,當(dāng)前指令指針(PC),寄存器集合和堆棧組成。另外,線程是進(jìn)程中的一個實體,是被系統(tǒng)獨立調(diào)度和分派的基本單位
    發(fā)表于 12-01 13:42 ?1589次閱讀

    java的線程安全、單例模式、JVM內(nèi)存結(jié)構(gòu)

    線程安全就是多線程訪問時,采用了加鎖機(jī)制,當(dāng)一個線程訪問類的某個數(shù)據(jù)時,進(jìn)行保護(hù),其他線程不能進(jìn)行訪問直到該
    發(fā)表于 03-12 10:30 ?0次下載

    什么是線程安全 如何實現(xiàn)線程安全代碼

    相信有很多同學(xué)在面對多線程代碼時都會望而生畏,認(rèn)為多線程代碼就像一頭難以馴服的怪獸,你制服不了這頭怪獸它就會反過來吞噬你。
    的頭像 發(fā)表于 05-17 12:45 ?1666次閱讀

    如何理解線程安全

    本次分享線程安全的基礎(chǔ)知識。
    的頭像 發(fā)表于 05-08 15:03 ?888次閱讀
    如何<b class='flag-5'>理解</b><b class='flag-5'>線程</b><b class='flag-5'>安全</b>?

    線程安全怎么辦

    線程安全一直是多線程開發(fā)中需要注意的地方,可以說,并發(fā)安全保證了所有的數(shù)據(jù)都安全。 1 線程
    的頭像 發(fā)表于 10-10 15:00 ?405次閱讀
    <b class='flag-5'>線程</b><b class='flag-5'>安全</b>怎么辦

    如何知道你的代碼是否線程安全

    在并發(fā)編程時,如果多個線程訪問同一資源,我們需要保證訪問的時候不會產(chǎn)生沖突,數(shù)據(jù)修改不會發(fā)生錯誤,這就是我們常說的 線程安全 。 那什么情況下,訪問數(shù)據(jù)時是安全的?什么情況下,訪問數(shù)據(jù)
    的頭像 發(fā)表于 11-01 11:42 ?776次閱讀
    如何知道你的代碼是否<b class='flag-5'>線程</b><b class='flag-5'>安全</b>

    redis多線程還能保證線程安全

    Redis是一種使用C語言編寫的高性能鍵值存儲系統(tǒng),它是單線程的,因為使用了多路復(fù)用的方式來處理并發(fā)請求。這樣的實現(xiàn)方式帶來了很好的性能,但同時也引發(fā)了一些線程安全方面的問題。 在Redis中,由于
    的頭像 發(fā)表于 12-05 10:28 ?1919次閱讀