1 由于多个线程使用同一个变量,各个线程
都对变量进行操作,那么变量的值会被不同线程操作覆盖。
通常 变量A <-- 线程A
<-- 线程B
TLS 变量A <-- 线程A
变量A <-- 线程B
2解决办法
2.1 使用关键字 __declspec(thread)
(声明该变量后,线程访问时自动创建变量副本,每个线程操作(读或写)自己的副本,无法操作实际变量)
__declspec(thread) CHAR * g_pszText2 = NULL;
2.2 TLS相关API()
(线程写入变量值之后,手动创建一个私有变量副本,可以对副本读取,但是不可以写入)2.2.1 创建TLS索引
DWORD TlsAlloc(VOID)返回一个TLS索引号
2.2.2 设置值
BOOL TlsSetValue(DWORD dwTlsIndex, //TLS索引
LPVOID lpTlsValue //保存的值
);
2.2.3 获取值
LPVOID TlsGetValue(DWORD dwTlsIndex //TLS索引
);
返回存放在索引内的值
2.2.4 释放
BOOL TlsFree(DWORD dwTlsIndex //TLS索引
);
示例代码说明:两个线程向同一个索引中设置不同值,然后再从同一个索引中可以取出自己设置的值,说明TLS索引对每个线程做了不同的备份
// ThreadTls.cpp : Defines the entry point for the console application.
//#include "stdafx.h"
#include "stdlib.h"
#include "windows.h"CHAR * g_pszText = NULL;
DWORD g_nTlsIndex = 0;void Print( )
{printf( "g_pszText: %s\n", g_pszText );//从TLS索引中获取值CHAR * pszText = (CHAR *)TlsGetValue( g_nTlsIndex );printf( "TLS: %s\n", pszText );
}DWORD WINAPI PrintProc( LPVOID pParam )
{CHAR * pszText = (CHAR *)pParam;g_pszText = (CHAR *)malloc( 100 );strcpy( g_pszText, pszText );//将值保存到TLS索引当中TlsSetValue( g_nTlsIndex, g_pszText );while( 1 ){Print( );Sleep( 1000 );}return 0;
}void Create( )
{HANDLE hThread = NULL;DWORD nThreadID = 0;CHAR szText1[] = "ThreadProc 1----------";hThread = CreateThread( NULL, 0,PrintProc, szText1, 0, &nThreadID );CHAR szText2[] = "-----ThreadProc 2-----";hThread = CreateThread( NULL, 0,PrintProc, szText2, 0, &nThreadID );WaitForSingleObject( hThread, INFINITE );
}int main(int argc, char* argv[])
{ //创建TLS索引号g_nTlsIndex = TlsAlloc( );//创建线程Create( );//释放索引TlsFree( g_nTlsIndex );return 0;
}