linux中线程局部存储(TLS)

本文介绍了如何在C语言中使用TLS(ThreadLocalStorage)机制为全局变量创建线程私有副本,避免资源竞争。通过GCC的__thread关键字,不同线程对同一TLS变量的操作互不干扰,确保数据独立性。
摘要由CSDN通过智能技术生成

TLS(Thread Local Storage)

很多时候,开发者在编写多线程程序的时候都希望存储一些线程私有的数据。我们知道术语每个线程私有的数据包括线程的栈和当前的寄存器,但时这两种存储都是非常不可靠的,栈会在每个函数退出和进入的时候被改变;而寄存器少得可怜,我们不可能拿寄存器区存储所需要得数据。如果我们要在线程中使用一个全局变量,但希望这个全局变量时线程私有得,而不是所以线程共享得,该怎么办呢?这个时候就须要用到线程局部存储这个机制了。TLS得用法很简单,如果要定义一个全局变量TLS类型得,只需要在它签名加上相应得关键字即可。

对于gcc来说,这个关键字就是__thread,比如我们定义一个TLS得全局整形变量:

__thread int number;

一旦一个全局变量被定义成TLS类型得,那么每个线程都会拥有这个变量得一个副本,任何线程对该变量得修改都不会影响其他线程中该变量得副本。

代码如下:

#include<pthread.h>
#include<stdio.h>

__thread int g_num = 1;
void funA(void)
{
        int i = 0;
        for (i = 0; i < 5000; i++)
                g_num++;
        printf("A g_num=%d, &g_num=%d\n", g_num, &g_num);
}
void funB(void)
{
        int i = 0;
        for (i = 0; i < 5000; i++)
                g_num++;
        printf("B g_num=%d, &g_num=%d\n", g_num, &g_num);
}

int main(int argc, char** argv)
{
        pthread_t p1, p2;
        if (pthread_create(&p1, NULL, funA, NULL) == -1)
        {
                printf("pthread_create A fail\n");
                return -1;
        }
        if (pthread_create(&p2, NULL, funB, NULL) == -1)
        {
                printf("pthread_create B fail\n");
                return -1;
        }
        pthread_join(p1, NULL);
        pthread_join(p2, NULL);
        printf("main g_num=%d, &g_num=%d\n", g_num, &g_num);
        return 0;
}

2核上运行结果如下:

A g_num=5001, &g_num=246437628
B g_num=5001, &g_num=238044924
main g_num=1, &g_num=246441788
可见地址都不一样,线程之间相互不影响

当去掉__thread关键字得2核上运行结果如下:

A g_num=6628, &g_num=119885840
B g_num=8482, &g_num=119885840
main g_num=8482, &g_num=119885840
可见地址一样,存在资源竞争问题

  • 6
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值