青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

桃源谷

心靈的旅行

人生就是一場旅行,不在乎旅行的目的地,在乎的是沿途的風景和看風景的心情 !
posts - 32, comments - 42, trackbacks - 0, articles - 0
  C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理

準則3:多線程程序里不準使用fork

Posted on 2008-06-01 20:16 lymons 閱讀(13107) 評論(0)  編輯 收藏 引用 所屬分類: C++CUnix/Linux文章翻譯
From 2008精選

鉄則3: マルチスレッドのプログラムでのforkはやめよう
準則3:多線程程序里不準使用fork

 

マルチスレッドのプログラムで、「自スレッド以外のスレッドが存在している狀態」でfork

 

何が起きるか
能引起什么問題呢?

 

実例から見てみましょう。次のコードを実行すると、子プロセスは実行開始直後のdoit() 呼び出し時、高い確率でデッドロックします。
那看看實例吧.一執行下面的代碼,在子進程的執行開始處調用doit(),發生死鎖的機率會很高.

 

 1void* doit(void*{
 2
 3    static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
 4
 5    pthread_mutex_lock(&mutex);
 6
 7    struct timespec ts = {100}; nanosleep(&ts, 0); // 10秒寢る
 8                                                     // 睡10秒
 9
10    pthread_mutex_unlock(&mutex);
11
12    return 0;
13
14}

15
16 
17
18int main(void{
19
20pthread_t t;  
21
22pthread_create(&t, 0, doit, 0); // サブスレッド作成?起動
23
24                                // 做成并啟動子線程
25
26    if (fork() == 0{
27
28        // 子プロセス。
29
30        // 子プロセスが生成される瞬間、親のサブスレッドはnanosleep中の場合が多い。
31
32        //子進程
33
34        //在子進程被創建的瞬間,父的子進程在執行nanosleep的場合比較多
35
36        doit(0); return 0;
37
38    }

39
40pthread_join(t, 0); // サブスレッド完了待ち
41
42                    // 等待子線程結束
43
44}

45

 

以下にデッドロックの理由を説明いたします。
以下是說明死鎖的理由.

 

一般に、forkを行うと
一般的,fork做如下事情

  1. 親プロセスの「データ領域」は子プロセスにそのままコピー
  2. 子プロセスは、シングルスレッド狀態で生成
  1. 父進程的內存數據會原封不動的拷貝到子進程中
  2. 子進程在單線程狀態下被生成

されます。データ領域には、靜的記憶域を持つ変數*2が格納されていますが、それらは子プロセスにコピーされます。また、親プロセスにスレッドが複數存在していても、子プロセスにそれらは継承されません。forkに関する上記2つの特徴がデッドロックの原因となります。
在內存區域里,靜態變量*2mutex的內存會被拷貝到子進程里.而且,父進程里即使存在多個線程,但它們也不會被繼承到子進程里. fork的這兩個特征就是造成死鎖的原因.
譯者注: 死鎖原因的詳細解釋 ---
    1.
線程里的doit()先執行.
    2. doit
執行的時候會給互斥體變量mutex加鎖.
    3. mutex
變量的內容會原樣拷貝到fork出來的子進程中(在此之前,mutex變量的內容已經被線程改寫成鎖定狀態).
    4.
子進程再次調用doit的時候,在鎖定互斥體mutex的時候會發現它已經被加鎖,所以就一直等待,直到擁有該互斥體的進程釋放它(實際上沒有人擁有這個mutex).
    5.
線程的doit執行完成之前會把自己的mutex釋放,但這是的mutex和子進程里的mutex已經是兩份內存.所以即使釋放了mutex鎖也不會對子進程里的mutex造成什么影響.

 

例えば次のようなシナリオを考えてみてください。上記のマルチスレッドプログラムでの不用意なforkによって子プロセスがデッドロックすることがわかると思います*3
例如,請試著考慮下面那樣的執行流程,就明白為什么在上面多線程程序里不經意地使用fork就造成死鎖了*3.

1.    fork前の親プロセスでは、スレッド12が動いている

2.    スレッド1doit関數を呼ぶ

3.    doit関數が自身のmutexをロックする

4.    スレッド1nanosleepを実行し、寢る

5.    ここで処理がスレッド2に切り替わる

6.    スレッド2fork関數を呼ぶ

7.    子プロセスが生成される。

8.    この時、子プロセスのdoit関數用mutexは「ロック狀態」である。また、ロック狀態を解除するスレッドは子プロセス中には存在しない!

9.    子プロセスが処理を開始する。

10.        子プロセスがdoit関數を呼ぶ

11.        子プロセスがロック済みのmutexを再ロックしてしまい、デッドロックする

1.    fork前的父進程中,啟動了線程12

2.    線程1調用doit函數

3.    doit函數鎖定自己的mutex

4.    線程1執行nanosleep函數睡10

5.    在這兒程序處理切換到線程2

6.    線程2調用fork函數

7.    生成子進程

8.    這時,子進程的doit函數用的mutex處于鎖定狀態”,而且,解除鎖定的線程在子進程里不存在

9.    子進程的處理開始

10.子進程調用doit函數

11.子進程再次鎖定已經是被鎖定狀態的mutex,然后就造成死鎖

このdoit関數のように、マルチスレッド下でのforkで問題を引き起こす関數を、「fork-unsafeな関數」と呼ぶことがあります。逆に、問題を起こさない関數を「fork-safeな関數」と呼ぶことがあります。一部の商用UNIX*4では、OSの提供する関數について、ドキュメントにfork-safetyの記載がありますが、Linux(glibc)にはもちろん! 記載がありません。POSIXでも特に規定がありませんので、どの関數がfork-safeであるかは殆ど判別不能です。わからなければunsafeと考えるほうが良いでしょう。 (2004/9/12 追記) Wolfram Glogerさんが非同期シグナルセーフな関數を呼ぶのは規格準拠と言っておられるので調べてみたら、pthread_atforkのところ "In the meantime*5, only a short list of async-signal-safe library routines are promised to be available." とありました。そういうことのようです。
像這里的doit函數那樣的,在多線程里因為fork而引起問題的函數,我們把它叫做”fork-unsafe函數”.反之,不能引起問題的函數叫做”fork-safe函數”.雖然在一些商用的UNIX,源于OS提供的函數(系統調用),在文檔里有fork-safety的記載,但是在Linux(glibc)里當然!不會被記載.即使在POSIX里也沒有特別的規定,所以那些函數是fork-safe,幾乎不能判別.不明白的話,作為unsafe考慮的話會比較好一點吧.(2004/9/12追記)Wolfram Gloger說過,調用異步信號安全函數是規格標準,所以試著調查了一下,pthread_atfork這個地方里有” In the meantime*5, only a short list of async-signal-safe library routines are promised to be available.”這樣的話.好像就是這樣.

 

ちなみに、malloc関數は自身に固有のmutexを持っているのが通例ですので、普通はfork-unsafeです。malloc関數に依存する數多くの関數、例えばprintf関數などもfork-unsafeとなります。
隨便說一下,malloc函數就是一個維持自身固有mutex的典型例子,通常情況下它是fork-unsafe.依賴于malloc函數的函數有很多,例如printf函數等,也是變成fork-unsafe.

いままでthread+forkは危険と書いてきましたが、一つだけ特例があります。「fork直後にすぐexecする場合は、特例として問題がない」のです。何故でしょう..exec系関數*6が 呼ばれると、プロセスの「データ領域」は一旦綺麗な狀態にリセットされます。したがって、マルチスレッド狀態のプロセスであっても、fork後にすぐ、危 険な関數を一切呼ばずにexec関數を呼べば、子プロセスが誤動作することはないのです。ただし、「すぐ」と書いてあることに注意してください。exec前に printf(“I’m child process”); を一発呼ぶだけでもデッドロックの危険があります!
直到目前為止,已經寫上了thread+fork是危險的,但是有一個特例需要告訴大家.”fork后馬上調用exec的場合,是作為一個特列不會產生問題的”. 什么原因呢..? exec函數*6一被調用,進程的內存數據就被臨時重置成非常漂亮的狀態.因此,即使在多線程狀態的進程里,fork后不馬上調用一切危險的函數,只是調用exec函數的話,子進程將不會產生任何的誤動作.但是,請注意這里使用的馬上這個詞.即使exec前僅僅只是調用一回printf(“I’m child process”),也會有死鎖的危險.
譯者注:exec函數里指明的命令一被執行,改命令的內存映像就會覆蓋父進程的內存空間.所以,父進程里的任何數據將不復存在.

 

災いをどう回避するか
如何規避災難呢?

 

マルチスレッドのプログラムでのforkを安全に行うための、デッドロック問題回避の方法はあるでしょうか?いくつか考えてみます。
為了在多線程的程序中安全的使用fork,而規避死鎖問題的方法有嗎?試著考慮幾個.

 

回避方法1: forkを行う場合は、それに先立って他スレッドを全て終了させる
規避方法1:fork的時候,在它之前讓其他的線程完全終止.

 

forkに先立って他スレッドを全て終了させておけば、問題はおきません。ただ、それが可能なケースばかりではないでしょう。また、何らかの要因で他スレッドの終了が行われないままforkしてしまった場合、解析困難な不具合して問題が表面化してしまいます。
fork之前,讓其他的線程完全終止的話,則不會引起問題.但這僅僅是可能的情況.還有,因為一些原因而其他線程不能結束就執行了fork的時候,就會是產生出一些解析困難的不具合的問題.

 

回避方法2: fork直後に子プロセスがexecを呼ぶようにする
規避方法2:fork后在子進程中馬上調用exec函數

(2004/9/11 書き忘れていたので追記)
(2004/9/11
追記一些忘了寫的東西)

 

回 避方法1が取れない場合は、子プロセスはfork直後に、どんな関數(printfなどを含む)も呼ばずにすぐにexeclなど、execファミリーの関 數を呼ぶようにします。もし、"execしないfork"を一切使わないプログラムであれば、現実的な回避方法でしょう。
不用使用規避方法1的時候,fork后不調用任何函數(printf)就馬上調用execl,exec系列的函數.如果在程序里不使用沒有execfork”的話,這應該就是實際的規避方法吧.
譯者注:筆者的意思可能是把原本子進程應該做的事情寫成一個單獨的程序,編譯成可執行程序后由exec函數來調用.

 

回避方法3: 「他スレッド」ではfork-unsafeな処理を一切行わない
規避方法3:”其他線程,不做fork-unsafe的處理

 

forkを呼ぶスレッドを除く全てのスレッドが、fork-unsafeな処理を一切行わない方法です。數値計算の速度向上目的でスレッドを使用している場合*7などは、なんとか可能かもしれませんが、一般のアプリケーションでは現実的ではありません。どの関數がfork-safeなのか把握することだけでも容易ではないからです。fork-safeな関數、要するに非同期シグナルセーフな関數ですが、それは數えるほどしかないからです。この方法では malloc/new, printf すら使えなくなってしまいます。
除了調用fork的線程,其他的所有線程不要做fork-unsafe的處理.為了提高數值計算的速度而使用線程的場合*7,這可能是fork-safe的處理,但是在一般的應用程序里則不是這樣的.即使僅僅是把握了那些函數是fork-safe,做起來還不是很容易的.fork-safe函數,必須是異步信號安全函數,而他們都是能數的過來的.因此,malloc/new,printf這些函數是不能使用的.

 

回避方法4: pthread_atfork関數を用いて、fork前後に自分で用意したコールバック関數を呼んでもらう
規避方法4:使用pthread_atfork函數,即將fork之前調用事先準備的回調函數.

 

pthread_atfork 関數を用いて、fork前後に自分で用意したコールバック関數を呼んでもらい、コールバック內で、プロセスのデータ領域を掃除する方法です。しかし、OS 提供の関數(: malloc)については、コールバック関數から掃除する方法がありません。mallocの使用するデータ構造は外部からは見えないからです。よって、 pthread_atfork関數はあまり実用的ではありません。
使用pthread_atfork函數,在即將fork之前調用事先準備的回調函數,在這個回調函數內,協商清除進程的內存數據.但是關于OS提供的函數(:malloc),在回調函數里沒有清除它的方法.因為malloc里使用的數據結構在外部是看不見的.因此,pthread_atfork函數幾乎是沒有什么實用價值的.

 

回避方法5: マルチスレッドのプログラムでは、forkを一切使用しない
規避方法5:在多線程程序里,不使用fork

 

forkを一切使用しない方法です。forkするのではなく、素直にpthread_createするようにします。これも、回避策2と同様に現実的な方法であり、推奨できます。
就是不使用fork的方法.即用pthread_create來代替fork.這跟規避策2一樣都是比較實際的方法,值得推薦.

 

*1:子プロセスを生成するシステムコール
*1:生成子進程的系統調用

*2:グローバル変數や関數內のstatic変數
*2:全局變量和函數內的靜態變量

*3Linuxを使用するのであれば、pthread_atfork関數のman pageを見るとよいです。この種のシナリオについて若干の解説があります
*3:如果使用Linux的話,查看pthread_atfork函數的man手冊比較好.關于這些流程都有一些解釋.

*4SolarisHP-UXなど
*4
SolarisHP-UX

*5forkexecするまでの間
*5:從fork后到exec執行的這段時間

*6≒execveシステムコール
*6≒execve系統調用

*7:四則演算しか行わないならfork-safe
*7
僅僅做四則演算的話就是fork-safe


原文地址:http://d.hatena.ne.jp/yupo5656/20040715/p1 

我的個人簡歷第一頁 我的個人簡歷第二頁
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲日韩第九十九页| 亚洲天堂网在线观看| 欧美日韩1区2区| 亚洲一区二区免费| 一本一本大道香蕉久在线精品| 国产精品久久久久一区二区三区 | 欧美一区免费| 国内一区二区三区在线视频| 你懂的亚洲视频| 欧美大片在线观看| 亚洲小少妇裸体bbw| 午夜精品久久久久久久99热浪潮| 伊人春色精品| 亚洲美女在线视频| 国产精品视频精品视频| 久久天天躁夜夜躁狠狠躁2022| 欧美sm视频| 亚洲欧美激情精品一区二区| 欧美一区二区三区婷婷月色| 亚洲精品久久在线| 一区二区三区免费网站| 国产综合色在线| 亚洲国产精品成人一区二区| 欧美日韩1080p| 久久精品中文字幕免费mv| 欧美 日韩 国产 一区| 亚洲无限av看| 久久久久国产精品www| 99国产精品久久久| 亚洲欧美在线播放| 亚洲欧洲美洲综合色网| 亚洲私人黄色宅男| 在线观看亚洲精品视频| 一区二区三区视频在线| 激情亚洲成人| 一本色道久久99精品综合| 黄网动漫久久久| 9久草视频在线视频精品| 韩国在线一区| 亚洲最新中文字幕| 精品999成人| 制服丝袜激情欧洲亚洲| 伊人男人综合视频网| 一区二区冒白浆视频| 亚洲成色999久久网站| 亚洲婷婷综合久久一本伊一区| 亚洲高清一二三区| 亚洲一级在线观看| 亚洲日韩第九十九页| 欧美尤物巨大精品爽| 亚洲网站视频福利| 久久女同互慰一区二区三区| 亚洲一区三区视频在线观看| 免费观看日韩| 久久久91精品国产| 欧美日韩亚洲高清| 欧美成人一区二区在线| 国产欧美精品在线观看| 亚洲精品日韩一| 亚洲高清在线观看| 亚洲欧美日韩在线播放| 在线综合+亚洲+欧美中文字幕| 久久久人成影片一区二区三区观看| 午夜欧美不卡精品aaaaa| 欧美黄色免费| 欧美/亚洲一区| 国产午夜精品全部视频在线播放| 日韩五码在线| 亚洲国产日韩欧美综合久久| 欧美一区二区三区四区在线 | 久久久国产精彩视频美女艺术照福利 | 韩曰欧美视频免费观看| 一区二区欧美国产| 亚洲巨乳在线| 久久亚洲影音av资源网| 久久久精品性| 国产精品视频999| 一区二区三区 在线观看视频| 亚洲精品乱码久久久久久日本蜜臀| 久久久久99精品国产片| 欧美在线视频二区| 国产精品igao视频网网址不卡日韩| 亚洲国产美女精品久久久久∴| 在线看国产日韩| 久久成人精品无人区| 欧美在线视频播放| 国产精品日韩在线| 一区二区三区久久网| 一区二区三区产品免费精品久久75| 欧美国产日韩一区二区在线观看| 欧美激情国产高清| 黄色日韩精品| 久久精品视频va| 久久理论片午夜琪琪电影网| 国产一区二区成人| 亚洲欧美日韩精品久久奇米色影视| 亚洲欧美一区在线| 国产精品国产三级国产专区53| 在线一区二区三区四区五区| 亚洲制服少妇| 国产精品成人av性教育| 一区二区三区免费看| 亚洲免费在线观看| 国产精品久久久久9999高清| 亚洲私人影吧| 欧美一区二区三区四区在线观看地址 | 国语自产精品视频在线看| 先锋影音网一区二区| 久久高清国产| 国内精品久久久久久 | 久久综合色综合88| 狠狠入ady亚洲精品| 久久久久久9| 欧美成人精品一区| 亚洲级视频在线观看免费1级| 男人的天堂亚洲在线| 亚洲国语精品自产拍在线观看| 日韩亚洲欧美综合| 欧美啪啪一区| 亚洲视频精选| 久久精品国产77777蜜臀| 国模私拍视频一区| 久久久久免费视频| 亚洲成在人线av| 在线视频你懂得一区| 国产精品毛片高清在线完整版| 午夜精品久久久久久久99樱桃| 久久久精彩视频| 亚洲第一区色| 欧美精品18+| 亚洲无线观看| 久久免费国产精品| 亚洲第一在线| 欧美日韩裸体免费视频| 亚洲一区二区成人在线观看| 久久精品一区二区三区四区 | 一区二区三区产品免费精品久久75 | 亚洲国产日韩欧美在线动漫| 在线视频一区观看| 国产精品日韩欧美| 久久精选视频| 亚洲精品国产视频| 亚洲欧美国产视频| 狠狠色丁香婷综合久久| 欧美大胆成人| 在线视频你懂得一区| 久久精品一区二区国产| 亚洲精品免费在线播放| 欧美视频在线观看 亚洲欧| 午夜在线不卡| 亚洲国产精品黑人久久久| 亚洲淫片在线视频| 国产伊人精品| 欧美激情一区在线| 亚洲免费在线| 亚洲第一狼人社区| 亚洲欧美国产精品va在线观看| 在线观看亚洲a| 欧美午夜在线一二页| 久久精品国产综合| 亚洲精品一区二区三区不| 午夜一区不卡| 亚洲黄色毛片| 国产精品永久免费| 老鸭窝91久久精品色噜噜导演| 一区二区三区四区五区在线| 欧美99久久| 午夜精品久久久| 亚洲精品国产拍免费91在线| 国产精品自拍一区| 蜜臀av性久久久久蜜臀aⅴ| 亚洲无限av看| 91久久极品少妇xxxxⅹ软件| 久久国产精品网站| 99re6热只有精品免费观看| 国产亚洲一区精品| 欧美日韩www| 久久久99久久精品女同性| 中文精品在线| 欧美激情国产日韩| 久久精品国产v日韩v亚洲| 99视频超级精品| 伊人久久婷婷| 国产区在线观看成人精品| 欧美高清hd18日本| 欧美一区免费视频| 9人人澡人人爽人人精品| 欧美黄色精品| 久久久91精品国产一区二区三区| 亚洲一区二区三区四区五区黄| 亚洲国产精品视频| 国产日韩一区在线| 欧美午夜在线观看| 欧美大片一区二区| 久久久久在线| 午夜精品久久久久久久| 一区二区三区你懂的| 亚洲国产欧美在线人成| 媚黑女一区二区| 久久av二区|