【C语言编程】gcc关键字__thread

__thread

Thread Local Storage:线程局部存储(tls)分配的变量,每个当前线程有一个该变量的实例。__thread是gcc内置的线程局部存储设施,其存储效率可以和全局变量相比;__thread变量在每一个线程中都有一份独立实例,各线程值是互不干扰的。可以用来修饰那些带有全局性且值可能变,但是又不值得用全局变量保护的变量。只能修饰POD类型(类似整型指针的标量,不带自定义的构造、拷贝、赋值、析构的类型,二进制内容可以任意复制memset,memcpy,且内容可以复原)。
不能修饰class类型,因为无法自动调用构造和析构函数可用于修饰全局变量,函数内的静态变量,不能修饰函数的局部变量或class的普通成员变量。且__thread变量值只能初始化为编译器常量。

注意:
__thread限定符(specifier)可以单独使用,也可带有extern或static限定符,但不能带有其它存储类型的限定符。
__thread可用于全局的静态文件作用域,静态函数作用域或一个类中的静态数据成员。不能用于块作用域,自动或非静态数据成员。

1. 示例

#include <iostream>
#include <pthread.h>
#include <unistd.h>

const int i = 5;
__thread int var = i;//两种方式效果一样
static __thread int var2 = 15;

static void *worker1(void *arg) {
  std::cout << "worker1 var :" << ++var << std::endl;        // 6
  std::cout << "worker1 var addr :" << &var << std::endl;    // 0x7f73a20f86f4
  std::cout << "worker1 var2 :" << ++var2 << std::endl;      // 16
  std::cout << "worker1 var2 addr :" << &var2 << std::endl;  // 0x7f73a20f86f8
}

static void *worker2(void *arg) {
  sleep(1);//等待线程1改变var值,验证是否影响线程2
  std::cout << "worker2 var :" << --var << std::endl;       // 4
  std::cout << "worker2 var addr :" << &var << std::endl;    // 0x7f73a18f76f4
  std::cout << "worker2 var2 :" << --var2 << std::endl;      // 14
  std::cout << "worker2 var2 addr :" << &var2 << std::endl;  // 0x7f73a18f76f8
}

int main() {
  pthread_t pid1, pid2;
  static __thread int temp = 10;//修饰函数内的static变量

  pthread_create(&pid1, NULL, worker1, NULL);
  pthread_create(&pid2, NULL, worker2, NULL);
  pthread_join(pid1, NULL);
  pthread_join(pid2, NULL);

  std::cout << "main var addr :" << &var << std::endl;     // 0x7f73a30f0734
  std::cout << "main var2 addr :" << &var2 << std::endl;   // 0x7f73a30f0738
  std::cout << temp << std::endl;                          //输出10
  return 0;
}
 

输出:

worker1 var :6
worker1 var addr :0x6000034fc030
worker1 var2 :16
worker1 var2 addr :0x6000034fc038
worker2 var :4
worker2 var addr :0x6000034fc030
worker2 var2 :14
worker2 var2 addr :0x6000034fc038
main var addr :0x6000034f0000
main var2 addr :0x6000034f0008
10

参考资料

1. __thread详解(gcc关键字)

posted @ 2023-04-07 09:39  苏格拉底的落泪  阅读(320)  评论(0编辑  收藏  举报