首页 > 其他分享 >【GiraKoo】线程本地存储(Thread Local Storage, TLS)

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

时间:2024-10-30 15:45:11浏览次数:1  
标签:TLS GiraKoo Thread VOID lpvData TlsGetValue 线程 dwTlsIndex

【技术分享】线程本地存储(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;
}

标签:TLS,GiraKoo,Thread,VOID,lpvData,TlsGetValue,线程,dwTlsIndex
From: https://www.cnblogs.com/girakoo/p/18515955

相关文章

  • 【GiraKoo】常用编码的对比(ASCII,GB2312,GBK,GB18030,UCS,Unicode)
    甯哥敤缂栫爜鐨勫姣旓紙ASCII锛孏B2312锛孏BK锛孏B18030锛孶CS锛孶nicode锛�鍦ㄧ▼搴忓紑鍙戜腑锛屾枃瀛楃紪鐮佷竴鐩存壆婕旂潃浜虹暅鏃犲锛屽嵈鑳屽悗鎹呬竴鍒€鐨勮鑹层€�鍙兘鍦ㄦ簮浠g爜鏂囦欢涓紝娉ㄩ噴鑾悕鍏跺鍦板彉鎴愪簡涔辩爜銆�鍙兘鏄彂閫佺粰鍒......
  • 【GiraKoo】C++编译中常用的内置宏
    开源项目:https://girakoo.com/联系方式:[email protected]简介针对不同的平台,很多头文件,函数名称,类型占用空间不一致。为了保证跨平台可编译,经常需要在项目中使用宏进行区分系统宏操作系统可使用的宏Windows32位_WIN32Windows64位_WIN32;_WIN64Linux__linu......
  • 【GiraKoo】Android系统版本代号一览
    Android系统版本代号一览VersionCodeNameAPI时间13.0Tiramisu332022.0812.0SnowCone31,322021.1011.0R/11302020.0910.0Q/10292019.059.0P/(Pie)282018.088.1O_MR1272017.128.0O/(Oreo)262017.087.1N_MR125201......
  • 【GiraKoo】C++中static关键字的作用
    C++中static关键字的作用在程序中良好的使用static,const,private等关键字,对于代码的健壮性有很大的帮助。本文介绍的就是C++中static关键字的一些常见用法与区别。适合萌新程序员理解static的作用。一、在类内修饰变量限制变量的存储位置(保存在静态区),所有类对象共享一份数据。......
  • 【GiraKoo】C++多线程消息分发架构
    【开源需求】C++多线程消息分发架构项目【gi_messager】在多线程环境中,为每个线程提供独立的消息队列MessageLoop。注:主线程默认自动创建消息队列。MessageLoopCenter提供MessageLoop的查询功能。能够获得指定MessageLoop的句柄。同一个MessageLoop可以绑定多个......
  • .NET中的线程池ThreadPool(链接)
    微软推荐在.NET中使用多线程开发时,都使用线程池,下面这篇微软文档介绍了.NET中的线程池类ThreadPool:ThreadPoolClass注意上面文档中的这句话:Thereisonethreadpoolperprocess.也就是说,每个.NET进程(process)中有一个线程池,线程池在每个.NET进程中只有一个,一个.NET进程中......
  • 操作系统(7) (POSIX--Linux线程编程---使用多线程计算平方pthread_t/create/join应用)
    1.代码目的我们希望创建一个程序:启动多个线程,每个线程计算一个数字的平方值。每个线程将计算结果返回给主线程。主线程接收每个线程的返回值,并将结果打印出来。在这个例子中,我们通过传递不同的参数给每个线程,来让每个线程计算不同数字的平方值。2.代码实现以下是代码的......
  • ThreadLocal为什么会发生内存泄漏
    ThreadLocal会发生内存泄漏的原因有:一、ThreadLocal的生命周期与线程的生命周期相关;二、长时间不使用ThreadLocal可能导致内存泄漏等。ThreadLocal的生命周期与线程的生命周期相关是指,如果ThreadLocal没有被正确地清除,就会导致ThreadLocalMap中的Entry长时间无法被回收,从而导致内......
  • ScheduledThreadPoolExecutor的介绍与使用
    ScheduledThreadPoolExecutor是Java中的一个类,它继承自ThreadPoolExecutor,并实现了ScheduledExecutorService接口。这个类主要用于在给定的延迟之后或周期性地执行任务,是处理定时任务的一个强大工具。一、主要特点线程池大小固定:ScheduledThreadPoolExecutor的线程池大小......
  • 深入理解ThreadLocal底层原理
    ThreadLocal是线程私有的,各个线程之间是隔离的。可以想象一下每次线程创建的时候在堆上预先分配一个内存空间用于存储ThreadLocal的数据。(1)当线程被创建时,线程都会有一个成员变量ThreadLocalMap。//每个线程定义一个成员变量ThreadLocalMap。publicclassThreadimplemen......