这个原作者的这个地方写错了 且他举的例子非常不好。最后有我的修正版本

pthread_setspecific(key, (void *)&my_errno);

linux 线程私有数据之一键多值技术TSD池 2012-09-15 16:22:08

分类: LINUX

               linux 线程私有数据之 一键多值技术

进程内的所有线程共享进程的数据空间,因此全局变量为所有线程所共有。但有时线程也需要保存自己的私有数据,这时可以创建线程私有数据(Thread-
specific Date)TSD来解决。在线程内部,私有数据可以被各个函数访问,但对其他线程是屏蔽的。一个明显的例子是errno,每个线程都有自己的副本,不然由于线程间的切换,在一个线程里输出的很可能是令一线程的出错信息。

线程私有数据采用了一种一键多值的技术,即一个键对应多个数值。访问数据时都是通过键值来访问,好像是对一个变量进行访问,其实是在访问不同的数据。使用
线程私有数据时,首先要为每个线程数据创建一个相关联的键。POSIX中操作线程私有数据的主要通过以下4个函数来实现:pthread_key_create(创建一个键),pthread_setspecific(为一个键设置线程私有数据),pthread_getspecific(从一个键读取线程私有数据),pthread_key_delete(删除一个键)。这几个函数的声明如下:

#include <pthread.h>

int pthread_key_create(pthread_key_t *key, void (*destr_function) (void*));

int pthread_key_delete(pthread_key_t key);

int pthread_setspecific(pthread_key_t key, const void *pointer);

void * pthread_getspecific(pthread_key_t key);
   
   pthread_key_create:从Linux的TSD池中分配一项,将其值赋给key供以后访问使用,它的第一个参数key为指向键值的指针,第二个参数为一个函数指针,如果指针不为空,则在线程退出时将以key所关联的数据为参数调用destr_function(),释放分配的缓冲区。
    key一旦被创建,所有线程都可以访问它,但各线程可以根据自己的需要往key中填入不同的值,这就相当于提供了一个同名而不同值的全局变量,一键多值。一键多值靠的是一个关键数据结构数组,即TSD池其结构如下:

static struct pthread_key_struct pthread_keys[PTHREAD_KEYS_MAX] ={{0,NULL}};

创建一个TSD就相当于将结构数组中的某一项设置为“in_use”,并将其索引返回给*key,然后设置destructor函数
destr_function。pthread_key_create 创建一个新的线程特定数据Key时,系统搜索其所在进程的 Key
结构数组,找出其中第一个不在使用的元素,并返回该元素的索引键。
    pthread_setspecific:该函数将pointer的值(不是内容)与key相关联。用pthread_setspecific为一个键指定新的线程数据时,线程必须先释放原有的线程数据用以回收空间。
    pthread_getspecific:通过该函数得到与key相关联的数据。
    pthread_key_delete:该函数用来删除一个键,键所占用的内存将被释放。需要注意的是,键占用的内存被释放,与该键关联的线程数据所占用的内存并不被释放。因此,线程数据的释放必须在释放键之前完成。
    下面我们通过一个例子来学习使用这几个函数的使用,功能类似errno在线程的使用,我猜想全局变量errno是在创建一个线程pthread_create时利用TSD池为该线程分配一个errno的副本的。

点击(此处)折叠或打开

  1. /*   my_errno.c
  2. 编译链接:gcc my_errno.c -o my_errno -lpthread
  3. */
  4. // 线程私有数据,一键多值,tsd池
  5. #include <stdio.h>
  6. #include <string.h>
  7. #include <pthread.h>
  8. int my_errno = 0;
  9. pthread_key_t key;
  10. void print_errno(char *str)
  11. {
  12. printf("%s my_errno:%d\n",str, my_errno);
  13. }
  14. void *thread2(void *arg)
  15. {
  16. printf("thread2 %ld is running\n",pthread_self());
  17. pthread_setspecific(key, (void *)my_errno);
  18. printf("thread2 %ld returns %d\n",pthread_self(),\
  19. pthread_getspecific(key));
  20. my_errno = 2;
  21. print_errno("thread2");
  22. }
  23. void *thread1(void *arg)
  24. {
  25. pthread_t thid2;
  26. printf("thread1 %ld is running\n",pthread_self());
  27. pthread_setspecific(key, (void *)my_errno);
  28. my_errno = 1;
  29. pthread_create(&thid2, NULL, thread2, NULL);
  30. sleep(2);
  31. printf("thread1 %ld returns %d\n",pthread_self(),\
  32. pthread_getspecific(key));
  33. print_errno("thread1");
  34. }
  35. void destr(void *arg)
  36. {
  37. printf("destroy memory\n");
  38. }
  39. int main(void)
  40. {
  41. pthread_t thid1;
  42. printf("main thread begins running. my_errno=%d\n",my_errno);
  43. pthread_key_create(&key, destr);
  44. pthread_create(&thid1, NULL, thread1, NULL);
  45. sleep(4);
  46. pthread_key_delete(key);
  47. printf("main thread exit\n");
  48. return 0;
  49. }

./my_errno 后你会发现虽然my_errno是全局的,但在thread1与thread2保留的是私有数据。

/*   my_errno.c
编译链接:gcc my_errno.c -o my_errno -lpthread
*/ // 线程私有数据,一键多值,tsd池
#include <stdio.h>
#include <string.h>
#include <pthread.h>
#include <errno.h> extern int errno ; pthread_key_t key; void print_errno(char *str)
{
printf("%s my_errno:%d\n",str, errno);
}
void *thread2(void *arg)
{
errno = ;
printf("thread2 %ld is running\n",pthread_self());
pthread_setspecific(key, (void *)&errno);
printf("thread2 %ld returns %d\n",pthread_self(),\
pthread_getspecific(key)); print_errno("thread2");
} void *thread1(void *arg)
{
errno = ;
pthread_t thid2; printf("thread1 %ld is running\n",pthread_self());
pthread_setspecific(key, (void *)&errno); pthread_create(&thid2, NULL, thread2, NULL);
sleep();
printf("thread1 %ld returns %d\n",pthread_self(),\
pthread_getspecific(key)); print_errno("thread1");
} void destr(void *arg)
{
printf("destroy memory\n");
} int main(void)
{
pthread_t thid1; printf("main thread begins running. my_errno=%d\n",errno);
pthread_key_create(&key, destr);
pthread_create(&thid1, NULL, thread1, NULL);
sleep();
pthread_key_delete(key); printf("main thread exit\n"); return ;
}

UE$ ./1
main thread begins running. my_errno=0
thread1 139672638236416 is running
thread2 139672629843712 is running
thread2 139672629843712 returns 293377696
thread2 my_errno:2
destroy memory

thread1 139672638236416 returns 301770400
thread1 my_errno:1
destroy memory

main thread exit

ZT linux 线程私有数据之 一键多值技术的更多相关文章

  1. 线程私有数据TSD——一键多值技术,线程同步中的互斥锁和条件变量

    一:线程私有数据: 线程是轻量级进程,进程在fork()之后,子进程不继承父进程的锁和警告,别的基本上都会继承,而vfork()与fork()不同的地方在于vfork()之后的进程会共享父进程的地址空 ...

  2. linux线程私有数据---TSD池

    进程内的所有线程共享进程的数据空间,所以全局变量为所有线程共有.在某些场景下,线程需要保存自己的私有数据,这时可以创建线程私有数据(Thread-specific Data)TSD来解决.在线程内部, ...

  3. Linux系统编程——线程私有数据

    在多线程程序中.常常要用全局变量来实现多个函数间的数据共享.因为数据空间是共享的,因此全局变量也为全部线程共同拥有. 測试代码例如以下: #include <stdio.h> #inclu ...

  4. pthread线程私有数据

    进程内所有的线程共享地址空间,任何声明为静态或外部的变量,或在进程堆声明的变量都可以被进程内的所有线程读写. static,extern,或堆变量的值是上次线程改写的值 一个线程真正拥有的唯一私有存储 ...

  5. linux多线程学习笔记六--一次性初始化和线程私有数据【转】

    转自:http://blog.csdn.net/kkxgx/article/details/7513278 版权声明:本文为博主原创文章,未经博主允许不得转载. 一,一次性初始化 以保证线程在调用资源 ...

  6. [转] unix/linux下线程私有数据实现原理及使用方法

     在维护每个线程的私有数据的时候,我们可能会想到分配一个保存线程数据的数组,用线程的ID作为数组的索引来实现访问,但是有一个问题是系统生成的线程 ID不能保证是一个小而连续的整数,并且用数组实现的时候 ...

  7. UNIX环境高级编程——线程私有数据

    线程私有数据(Thread-specific data,TSD):存储和查询与某个线程相关数据的一种机制. 在进程内的所有线程都共享相同的地址空间,即意味着任何声明为静态或外部变量,或在进程堆声明的变 ...

  8. Thread-specific data(TSD)线程私有数据

    Thread-specific data(TSD)线程私有数据 http://blog.chinaunix.net/uid-26885237-id-3209913.html linux多线程编程中引入 ...

  9. 线程私有数据和pthread_once

    #include <stdio.h> #include <pthread.h> pthread_key_t key; pthread_once_t ponce = PTHREA ...

随机推荐

  1. java.utils.UUID类介绍

    1 UUID介绍 UUID (Universally Unique Identifier)缩写,即通用唯一识别码,也是被开源软件基金会 (Open Software Foundation, OSF) ...

  2. 【CSS】 布局之多列等高

    这两天看了不少文章,对于css布局多了一些理解,现在来总结下. 我们来写一个最普遍的Top.Left.Content.Right.Foot布局. 第一步:自然是写一个坯子 <!DOCTYPE H ...

  3. 有意思的shader案例

    屏幕水波效果 https://blog.csdn.net/puppet_master/article/details/52975666

  4. SpringMVC的参数绑定

    一.@RequestMapping注解说明   通过@RequestMapping注解可以定义不同的处理器映射规则. URL路径映射 @RequestMapping(value="/item ...

  5. [PY3]——找出一个序列中出现次数最多的元素/collections.Counter 类的用法

    问题 怎样找出一个序列中出现次数最多的元素呢? 解决方案 collections.Counter 类就是专门为这类问题而设计的, 它甚至有一个有用的 most_common() 方法直接给了你答案 c ...

  6. bzoj 5368: [Pkusc2018]真实排名

    Description 小C是某知名比赛的组织者,该比赛一共有n名选手参加,每个选手的成绩是一个非负整数,定义一个选手的排名是 :成绩不小于他的选手的数量(包括他自己).例如如果3位选手的成绩分别是[ ...

  7. Dell解决黑苹果网卡(BCM94352ZAE/DW1560)怎么都打不开WiFi

    Dell解决黑苹果网卡(BCM94352ZAE/DW1560)怎么都打不开WiFi 2017年10月20日17:41:00 by SemiconductorKING 本来觉得驱动这个网卡不是个问题,以 ...

  8. nodejs时间工具类

    /** * * @fmt 格式化字符串 * @Date 为需要格式化的日期 * * 示例:format(new Date(),'yyyy-MM-dd hh:mm:ss'); * 返回值为字符串 */ ...

  9. git必会必知

    1 前言 git前身是BitKeeper,但是他不是开源软件,不符合当时开源趋势,于是就会有了开源的git,git开发只用了十天时间.目前git是公司开发必不可少的一个工具,用于多人开发的分布式版本控 ...

  10. 将一个数据库中表的数据导入另一个数据库(DB2)

    将一个数据库中的数据导入另一个数据库(DB2) 我这里举得例子是使用的DB2数据库,其他数据库思路也是这样啦! 1.从db2 数据库中将表中的数据导入本地的excel中 export to d:\my ...