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

統計

  • 隨筆 - 50
  • 文章 - 42
  • 評論 - 147
  • 引用 - 0

留言簿(6)

隨筆分類

文章分類

Link

搜索

  •  

積分與排名

  • 積分 - 167517
  • 排名 - 159

最新評論

閱讀排行榜

評論排行榜

使用線程局部存儲TLS

Thread local storage (TLS)統一進程的多個線程可以通過由TlsAlloc方法返回的索引值在線程自身的空間內存儲和取回一個值。在以下這個例子里,索引值在進程開始時創建,當各個線程啟動時,會各自申請一塊動態內存并且將內存指針通過TlsSetValue方法存儲到各自的TLS空間中(由先前的索引值標定)。CommonFunc方法使用TlsGetValue方法通過索引取得數據指針。在各個線程結束前,釋放動態內存塊。在進程結束見,調用TlsFree方法釋放索引。

 1#include <windows.h> 
 2#include <stdio.h> 
 3 
 4#define THREADCOUNT 4 
 5DWORD dwTlsIndex; 
 6 
 7VOID ErrorExit(LPSTR); 
 8 
 9VOID CommonFunc(VOID) 
10
11   LPVOID lpvData; 
12 
13// Retrieve a data pointer for the current thread. 
14 
15   lpvData = TlsGetValue(dwTlsIndex); 
16   if ((lpvData == 0&& (GetLastError() != ERROR_SUCCESS)) 
17      ErrorExit("TlsGetValue error"); 
18 
19// Use the data stored for the current thread. 
20 
21   printf("common: thread %d: lpvData=%lx\n"
22      GetCurrentThreadId(), lpvData); 
23 
24   Sleep(5000); 
25}
 
26 
27DWORD WINAPI ThreadFunc(VOID) 
28
29   LPVOID lpvData; 
30 
31// Initialize the TLS index for this thread. 
32 
33   lpvData = (LPVOID) LocalAlloc(LPTR, 256); 
34   if (! TlsSetValue(dwTlsIndex, lpvData)) 
35      ErrorExit("TlsSetValue error"); 
36 
37   printf("thread %d: lpvData=%lx\n", GetCurrentThreadId(), lpvData); 
38 
39   CommonFunc(); 
40 
41// Release the dynamic memory before the thread returns. 
42 
43   lpvData = TlsGetValue(dwTlsIndex); 
44   if (lpvData != 0
45      LocalFree((HLOCAL) lpvData); 
46 
47   return 0
48}
 
49 
50int main(VOID) 
51
52   DWORD IDThread; 
53   HANDLE hThread[THREADCOUNT]; 
54   int i; 
55 
56// Allocate a TLS index. 
57 
58   if ((dwTlsIndex = TlsAlloc()) == TLS_OUT_OF_INDEXES) 
59      ErrorExit("TlsAlloc failed"); 
60 
61// Create multiple threads. 
62 
63   for (i = 0; i < THREADCOUNT; i++
64   
65      hThread[i] = CreateThread(NULL, // default security attributes 
66         0,                           // use default stack size 
67         (LPTHREAD_START_ROUTINE) ThreadFunc, // thread function 
68         NULL,                    // no thread function argument 
69         0,                       // use default creation flags 
70         &IDThread);              // returns thread identifier 
71 
72   // Check the return value for success. 
73      if (hThread[i] == NULL) 
74         ErrorExit("CreateThread error\n"); 
75   }
 
76 
77   for (i = 0; i < THREADCOUNT; i++
78      WaitForSingleObject(hThread[i], INFINITE); 
79 
80   TlsFree(dwTlsIndex);
81
82   return 0
83}
 
84 
85VOID ErrorExit (LPSTR lpszMessage) 
86
87   fprintf(stderr, "%s\n", lpszMessage); 
88   ExitProcess(0); 
89}

90

常用情景:
各個線程所處理的對象有所不同,但是所需要的處理卻可能類似,例如多個線程同時處理多個文件,就可以將文件句柄存在在相應的Tls中,在使用相同的接口進行處理
背景知識:
每個線程除了共享進程的資源外還擁有各自的私有資源:一個寄存器組(或者說是線程上下文);一個專屬的堆棧;一個專屬的消息隊列;一個專屬的Thread Local Storage(TLS);一個專屬的結構化異常處理串鏈。
TLS 是一個良好的Win32 特質,讓多線程程序設計更容易一些。TLS 是一個機制,經由它,程序可以擁有全域變量,但在不同的線程里有不同的值。也就是說,進程中的所有線程都可以擁有全域變量,但這些變量其實是特定對某個線程才有意義。例如,你可能有一個多線程程序,每一個線程都對不同的文件寫文件(也因此它們使用不同的文件handle)。這種情況下,把每一個線程所使用的文件handle 儲存在TLS 中,將會十分方便。當線程需要知道所使用的handle,它可以從TLS 獲得。重點在于:線程用來取得文件handle 的那一段碼在任何情況下都是相同的,而從TLS中取出的文件handle 卻各不相同。非常靈巧,不是嗎?有全域變數的便利,卻又分屬各線程。

 

  雖然TLS 很方便,它并不是毫無限制。在Windows NT 和Windows 95 之中,有64 個DWORD slots 供每一個線程使用。這意思是一個進程最多可以有64 個「對各線程有不同意義」的DWORDs。 雖然TLS 可以存放單一數值如文件handle,更常的用途是放置指針,指向線程的私有資料。有許多情況,多線程程序需要儲存一堆數據,而它們又都是與各線程相關。許多程序員對此的作法是把這些變量包裝為C 結構,然后把結構指針儲存在TLS 中。當新的線程誕生,程序就配置一些內存給該結構使用,并且把指針儲存在為線程保留下來的TLS 中。一旦線程結束,程序代碼就釋放所有配置來的區塊。既然每一個線程都有64 個slots 用來儲存線程自己的數據,那么這些空間到底打哪兒來?在線程的學習中我們可以從結構TDB中看到,每一個thread database 都有64 個DWORDs 給TLS 使用。當你以TLS 函式設定或取出數據,事實上你真正面對的就是那64 DWORDs。好,現在我們知道了原來那些“對各線程有不同意義的全局變量”是存放在線程各自的TDB中阿。
 
    接下來你也許會問:我怎么存取這64個DWORDS呢?我又怎么知道哪個DWORDS被占用了,哪個沒有被占用呢?首先我們要理解這樣一個事實:系統之所以給我們提供TLS這一功能,就是為了方便的實現“對各線程有不同意義的全局變量”這一功能;既然要達到“全局變量”的效果,那么也就是說每個線程都要用到這個變量,既然這樣那么我們就不需要對每個線程的那64個DWORDS的占用情況分別標記了,因為那64個DWORDS中的某一個一旦占用,是所有線程的那個DWORD都被占用了,于是KERNEL32 使用兩個DWORDs(總共64 個位)來記錄哪一個slot 是可用的、哪一個slot 已經被用。這兩個DWORDs 可想象成為一個64 位數組,如果某個位設立,就表示它對應的TLS slot 已被使用。這64 位TLS slot 數組存放在process database 中(在進程一節中的PDB結構中我們列出了那兩個DWORDs)。
 
下面的四個函數就是對TLS進行操作的:

  (1)TlsAlloc  

上面我們說過了KERNEL32 使用兩個DWORDs(總共64 個位)來記錄哪一個slot 是可用的、哪一個slot 已經被用。當你需要使用一個TLS slot 的時候,你就可以用這個函數將相應的TLS slot位置1。 

 (2)TlsSetValue  

TlsSetValue 可以把數據放入先前配置到的TLS slot 中。兩個參數分別是TLS slot 索引值以及欲寫入的數據內容。TlsSetValue 就把你指定的數據放入64 DWORDs 所組成的數組(位于目前的thread database)的適當位置中。  

 (3)TlsGetValue  

這個函數幾乎是TlsSetValue 的一面鏡子,最大的差異是它取出數據而非設定數據。和TlsSetValue 一樣,這個函數也是先檢查TLS 索引值合法與否。如果是,TlsGetValue 就使用這個索引值找到64 DWORDs 數組(位于thread database 中)的對應數據項,并將其內容傳回。  

 (4)TlsFree  

這個函數將TlsAlloc 和TlsSetValue 的努力全部抹消掉。TlsFree 先檢驗你交給它的索引值是否的確被配置過。如果是,它將對應的64 位TLS slots 位關閉。然后,為了避免那個已經不再合法的內容被使用,TlsFree 巡訪進程中的每一個線程,把0 放到剛剛被釋放的那個TLS slot 上頭。于是呢,如果有某個TLS 索引后來又被重新配置,所有用到該索引的線程就保證會取回一個0 值,除非它們再調用TlsSetValue。

posted on 2008-11-27 21:46 pear_li 閱讀(1214) 評論(0)  編輯 收藏 引用 所屬分類: windows kernel

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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区| 亚洲精品日韩综合观看成人91| 美女久久网站| 国产精品第13页| 亚洲成色最大综合在线| 国产精品一二一区| 亚洲精品一区二区三| 精品av久久久久电影| 亚洲欧美激情在线视频| 亚洲午夜av在线| 欧美黑人多人双交| 亚洲国产成人精品视频| 在线观看日韩精品| 久久久www免费人成黑人精品| 亚洲欧美日韩中文视频| 欧美日韩日本视频| 91久久视频| 日韩午夜激情电影| 欧美成人免费在线视频| 欧美激情一区二区三区四区| 亚洲国产成人精品视频| 久久影院午夜片一区| 裸体丰满少妇做受久久99精品| 国产一区再线| 久久www成人_看片免费不卡| 欧美一区二区精品| 国产日韩欧美综合一区| 午夜久久久久| 久久久久久一区二区| 国产一区二区三区成人欧美日韩在线观看| 亚洲天堂成人在线观看| 欧美一级黄色网| 国产一区二区三区无遮挡| 久久国产黑丝| 欧美α欧美αv大片| 亚洲激情av| 欧美国产精品久久| 99re6这里只有精品| 亚洲一区欧美二区| 国产情人综合久久777777| 久久国产色av| 欧美a一区二区| 日韩午夜在线播放| 欧美亚洲成人免费| 午夜精品影院| 欧美激情精品久久久久久黑人| 亚洲人成网站777色婷婷| 欧美激情二区三区| 亚洲一区三区在线观看| 久久亚洲一区二区| 亚洲精品在线免费观看视频| 欧美亚洲成人网| 久久精品99国产精品| 亚洲第一黄色网| 亚洲欧美激情四射在线日 | 欧美一区二区在线播放| 国产又爽又黄的激情精品视频| 久久精品一区蜜桃臀影院| 亚洲国语精品自产拍在线观看| 中文久久乱码一区二区| 国产自产2019最新不卡| 欧美激情视频在线免费观看 欧美视频免费一| 亚洲美女中文字幕| 久久精品免费看| 日韩一二三区视频| 国产人成一区二区三区影院| 欧美国产精品人人做人人爱| 午夜在线不卡| 亚洲日本视频| 久久综合九色综合久99| 在线视频精品一区| 亚洲成人在线观看视频| 国产精品久久久久久模特 | 午夜免费电影一区在线观看| 亚洲第一在线| 久久精品久久99精品久久| 日韩视频亚洲视频| 狠狠网亚洲精品| 国产精品国产三级国产普通话99| 久久久精品国产免大香伊 | 国产精品久久网| 欧美成人精品在线| 久久九九免费视频| 亚洲夜晚福利在线观看| 亚洲人成久久| 欧美不卡高清| 久久久国产91| 欧美亚洲视频在线看网址| av成人免费| 亚洲伦理网站| 亚洲国产欧美在线| 国语自产精品视频在线看一大j8 | 免费中文字幕日韩欧美| 久久精品成人| 性欧美18~19sex高清播放| 宅男噜噜噜66一区二区| 日韩视频久久| 亚洲日本国产| 亚洲茄子视频| 亚洲欧洲一区二区在线观看| 伊人一区二区三区久久精品| 国内精品久久国产| 国产原创一区二区| 韩国一区二区三区在线观看 | 亚洲激情网站| 亚洲国产精品久久久久婷婷老年| 国产综合香蕉五月婷在线| 国产日韩欧美二区| 国产欧美日韩亚洲| 国产一区二区三区高清| 国产一区二区在线观看免费| 国产亚洲一区二区在线观看| 国产一本一道久久香蕉| 狠狠色狠色综合曰曰| 在线观看国产欧美| 亚洲经典自拍| av成人天堂| 亚洲免费网址| 久久国产色av| 久久一区二区三区av| 蜜桃av一区二区三区| 欧美顶级少妇做爰| 亚洲激情在线视频| 一道本一区二区| 亚洲一区欧美| 久久免费黄色| 欧美精品videossex性护士| 欧美日韩国产小视频| 国产精品裸体一区二区三区| 国产一区二区无遮挡| 亚洲国产小视频| 亚洲一区精品电影| 久久精品国产亚洲aⅴ| 欧美成人精品一区| 亚洲另类自拍| 欧美一区二区免费观在线| 久久一区二区三区四区五区| 欧美激情一区在线观看| 国产精品日韩在线播放| 一区视频在线| 亚洲视频在线视频| 久久成人综合网| 亚洲国产黄色片| 亚洲影视中文字幕| 免费观看成人| 国产精品网站在线| 亚洲人成绝费网站色www| 亚洲欧美日本国产专区一区| 久久综合亚州| 一区二区三区四区五区精品视频| 欧美在线日韩精品| 欧美三级免费| 亚洲大胆女人| 性视频1819p久久| 亚洲激情国产精品| 欧美亚洲一区二区在线| 欧美激情综合五月色丁香| 国产在线不卡精品| 亚洲香蕉在线观看| 欧美激情国产日韩| 午夜亚洲一区| 欧美视频不卡| 亚洲精品乱码久久久久久黑人| 欧美中文字幕视频| 亚洲精品永久免费| 巨乳诱惑日韩免费av| 国产日本欧美一区二区三区在线| 日韩视频在线免费观看| 美女999久久久精品视频| 亚洲在线观看视频网站| 欧美精品成人| 亚洲成色777777在线观看影院| 欧美一区二区三区啪啪| 一本久久青青| 欧美精品一区二区久久婷婷 | 国产日韩av一区二区| 一区二区三区你懂的| 欧美福利一区二区| 久久成人人人人精品欧| 国产精品日日做人人爱| 一区二区欧美激情| 亚洲国产综合91精品麻豆| 久久久久久久综合色一本| 国产一区二区三区精品久久久| 亚洲影院在线观看| 日韩亚洲欧美一区二区三区| 免费试看一区| 最新国产乱人伦偷精品免费网站 | 亚洲一区二区影院| 欧美午夜免费影院| 亚洲一级特黄| 一区二区三区鲁丝不卡| 欧美三日本三级少妇三2023| 99在线精品观看| 亚洲精品久久久久中文字幕欢迎你 | 中文精品一区二区三区| 欧美视频精品在线观看| 亚洲性xxxx| 亚洲小说欧美另类社区| 国产精品亚洲综合色区韩国|