【GiraKoo】线程本地存储(Thread Local Storage, TLS)

【技术分享】线程本地存储(Thread Local Storage, TLS)

在项目开发中,遇到了关于TLS相关的问题。为了了解该机制的用途,在微软的官网查找了一些资料。
本文参考官方文档, 简单介绍一下TLS的用途与使用方法。

微软官方文档链接

一、简介

线程本地存储(TLS),可以使多个线程,通过TlsGetValue函数,获得各自线程独立的数据。

即,在进程中通过TlsAlloc,可以申请一个索引值(index)。
在不同的线程中,通过TlsSetValue/TlsGetValue,可以获得不同的数据。
注:TlsSetValue/TlsGetValue内部可能是通过线程ID进行了绑定,实现的功能。

二、官方示例(有调整)及个人注解

#include <windows.h>
#include <stdio.h>

#define THREADCOUNT 4
DWORD dwTlsIndex;

VOID ErrorExit(LPCSTR message);
VOID CommonFunc(VOID);
DWORD WINAPI ThreadFunc(VOID);

// main入口函数
int main(VOID)
{
    DWORD IDThread;
    HANDLE hThread[THREADCOUNT];
    int i;

    // 通过TlsAlloc函数,创建进程的tls索引,保存在dwTlsIndex
    if ((dwTlsIndex = TlsAlloc()) == TLS_OUT_OF_INDEXES)
    {
        // 如果返回TLS_OUT_OF_INDEXES,说明创建失败
        ErrorExit("TlsAlloc failed");
    }    

    // 通过for循环创建多个线程
    for (i = 0; i < THREADCOUNT; i++)
    {
        hThread[i] = CreateThread(NULL,                               // 默认security属性
                                  0,                                  // 默认stack size
                                  (LPTHREAD_START_ROUTINE)ThreadFunc, // 线程处理函数
                                  NULL,                               // no thread function argument
                                  0,                                  // use default creation flags
                                  &IDThread);                         // returns thread identifier

        // 判定创建线程结果
        if (hThread[i] == NULL)
            ErrorExit("CreateThread error\n");
    }

    // 等待所有线程执行完毕,退出循环
    for (i = 0; i < THREADCOUNT; i++)
        WaitForSingleObject(hThread[i], INFINITE);

    TlsFree(dwTlsIndex);

    return 0;
}

VOID ErrorExit(LPCSTR message)
{
    // 输出线程错误消息,并退出进程
    fprintf(stderr, "%s\n", message);
    ExitProcess(0);
}

VOID CommonFunc(VOID)
{
    LPVOID lpvData;

    // 取得dwTlsIndex指向的空间(此时获得的数据是各自线程通过TlsSetValue设置进去的)
    lpvData = TlsGetValue(dwTlsIndex);
    if ((lpvData == 0) && (GetLastError() != ERROR_SUCCESS))
        ErrorExit("TlsGetValue error");

    // 可以使用该空间数据
    printf("common: thread %d: lpvData=%lx\n",
           GetCurrentThreadId(), lpvData);

    Sleep(5000);
}

DWORD WINAPI ThreadFunc(VOID)
{
    LPVOID lpvData;

    // 申请动态空间,每个线程创建独自的空间。
    lpvData = (LPVOID)LocalAlloc(LPTR, 256);

    // 将动态空间绑定到Tls对应的Index空间上
    if (!TlsSetValue(dwTlsIndex, lpvData))
        ErrorExit("TlsSetValue error");

    printf("thread %d: lpvData=%lx\n", GetCurrentThreadId(), lpvData);

    // 通用处理函数,可以在内部使用TlsGetValue获得的数值
    CommonFunc();

    // 释放动态空间,每个线程创建独自的空间。
    lpvData = TlsGetValue(dwTlsIndex);
    
    if (lpvData != 0)
        LocalFree((HLOCAL)lpvData);

    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Windows下,C语言可以通过Windows API来实现线程本地存储Thread-Local StorageTLS)。以下是一个简单的示例代码来演示如何在Windows下使用C语言实现线程本地存储。 ```c #include <windows.h> // 声明线程本地存储变量 __declspec(thread) int tls_variable; // 线程函数 DWORD WINAPI ThreadFunction(LPVOID lpParam) { // 设置线程本地存储变量的值 tls_variable = 42; // 在线程内部访问线程本地存储变量的值 printf("Thread local variable value: %d\n", tls_variable); return 0; } int main() { HANDLE hThread; // 创建一个线程 hThread = CreateThread(NULL, 0, ThreadFunction, NULL, 0, NULL); if (hThread == NULL) { printf("Failed to create thread\n"); return 1; } // 等待线程结束 WaitForSingleObject(hThread, INFINITE); // 关闭线程句柄 CloseHandle(hThread); return 0; } ``` 在上面的示例代码中,我们使用`__declspec(thread)`关键字来声明一个线程本地存储变量`tls_variable`。在`ThreadFunction`线程函数中,我们设置了线程本地存储变量的值为42,并在线程内部打印了变量的值。 需要注意的是,使用`__declspec(thread)`关键字声明的线程本地存储变量只能是静态或全局变量,不能是局部变量。 编译运行上述代码后,你会看到输出中显示了线程本地存储变量的值为42。这证明了在线程之间,每个线程都有自己独立的线程本地存储空间。 这只是一个简单的示例,实际应用中,你可以根据需要在线程函数中使用线程本地存储来存储和访问特定于每个线程的数据。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

GiraKoo

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值