1. 程式人生 > >兩個執行緒同時呼叫一個函式會出現什麼情況?

兩個執行緒同時呼叫一個函式會出現什麼情況?

from: https://www.cnblogs.com/silentNight/p/5468805.html

最近在研究多執行緒,然後突然想到如果兩個執行緒同時訪問一個函式的話,要不要加鎖呢,加鎖怎麼加,不加又怎樣這樣的問題..然後去網上找了些帖子學習學習......

上面的程式碼明顯執行串了!!!!

函式本身只是程式碼,程式碼是隻讀的,無論多少個執行緒同時呼叫都無所謂,因為是隻讀嘛.但是函式裡面總要用到資料 ,如果資料屬性執行緒級別(比如函式形參-->區域性變數-->存在棧上-->每個執行緒都有自己的棧),那麼同時呼叫是沒關係的,因為用的都是本執行緒的資料;但是如果函式用到一些全域性資料,比如全域性變數,根據堆記憶體首地址去訪問的堆記憶體(形參傳入的),同時操作一個數據結構(如對一個連結串列有什麼操作),靜態區域性變數,那就不行了,必須要加鎖!!

 

http://blog.csdn.net/mq_ydn3102/article/details/8546722

1.執行緒中函式的呼叫

2.同一個程序中的兩個執行緒呼叫同一個處理函式

3.兩個不同的程序中的兩個執行緒呼叫同一個處理函式

 

1.執行緒中函式的呼叫:

每個程序都會有自己的虛擬地址空間,同一個程序中的每一個執行緒都會在這個虛擬空間中,並被分配相應的資源. 

執行緒中的函式實體實在記憶體中,而函式中的區域性變數則會一次被push到棧中.

如果是A函式呼叫B函式,則會先將A函式的引數和區域性變數push到棧中,再將B函式的引數和區域性變數push到棧中

具體如下(引數是從右往左壓入棧中,__stdcall和__cdecl呼叫都是這麼處理的):

void fun(param1, param2,. param3....paramN)

{

value1;

value2;

...

valueN;

}

如果執行上面的函式在棧中會這樣變化

1:將fun函式的傳入引數從右到左依次壓入到棧中

2:將fun函式的指標壓入棧中

3:將ebp,ebx等暫存器入棧儲存
4:在棧中依次為各個變數分配空間

執行後的處理:

1.釋放變數空間

2.彈出暫存器

3.ret,彈出eip,返回主調方程式碼

4.釋放引數空間

2.兩個執行緒呼叫同一個函式

 兩個執行緒中的函式的區域性變數由於是儲存在不同的執行緒中,因此不需要進行互斥處理(除非有非棧記憶體在搗亂,這種情況必須要有互斥鎖)

3.兩個不同程序中的兩個執行緒呼叫同一個處理函式

同樣,兩個執行緒中的函式的區域性變數由於是儲存在不同的執行緒中,因此不需要進行互斥處理

 

結論:

因此需要互斥處理的,一般是函式中有全域性變數,有動態申請的空間,有靜態區域性變數,有需要程序資料迴圈傳送(程序傳引數給執行緒)之類的操作需要進行互斥處理