系统程序员成长计划-内存管理(二)

news/2024/7/20 14:27:59 标签: 内存管理
系统程序员成长计划-内存管理(二)

转载时请注明出处和作者联系方式
文章出处:http://www.limodev.cn/blog
作者联系方式:李先静 <xianjimli at hotmail dot com>

线程局部存储(TLS)

同一个进程中的多个线程,它们的内存空间是共享的(栈除外),一个线程对内存的修改,对所有线程都有效。这是一个优点也是一个缺点。说它是优点,线程间的数据交换快捷高效。说它是缺点,一个线程死掉了,其它线程也性命不保。

在unix下,大家一直都对线程不太感兴趣,直到很晚以后才引入真正的线程。像X Sever要同时处理N个客户端的连接,每秒钟要响应上百万个请求,开发人员宁愿自己实现调度机制也不用线程。再如Apache(1.3x),在unix下的实现也是采用多进程的。

正如《unix编程艺术》中所说,线程局部存储的出现,使得这种情况出现了转机。采用线程局部存储,每个线程有一定的私有空间。这可以避免部分无意的破坏(当然无法避免有意的破坏行为),也省去线程访问共享数据时出现的问题。

其实这完全是因为unix程序不喜欢面向对象方法引起的,数据没有很好的封装起来,全局变量满天飞,在多线程情况下自然容易出问题。如果采用面向对象的方法,情况将会大为改观,而无需要线程局部存储来帮忙。

不过,多一种技术就多一种选择,知道线程局部存储肯定没有坏处。在有的情况下,没有线程局部存储,确实很难用其它办法实现。 比如,glibc会把最后一次操作的错误码记录在errno变量里,如果不采用线程局部存储,在多线程的情况下,当前线程取的errno可能不是自己上一 次操作的错误码。

线程局部存储在不同的平台有不同的实现,可移植性不太好。幸好要实现线程局部存储并不难,最简单的办法就是建立一个全局表,通过当前线程ID去查询相应的数据,因为各个线程的ID不同,查到的数据自然也不同了。

大多数平台都提供了线程局部存储的方法,无需要我们自己去实现,在Linux下有两种方法可以实现线程局部存储。

方法一: 使用pthread的函数

int pthread_key_create(pthread_key_t *key, void (*destructor)(void*));
int pthread_key_delete(pthread_key_t key);
void *pthread_getspecific(pthread_key_t key);
int pthread_setspecific(pthread_key_t key, const void *value);

示例:

static pthread_key_t key;
static pthread_once_t key_once = PTHREAD_ONCE_INIT;

static void make_key(void)
{
pthread_key_create(&key, NULL);

return;
}

void* thread_entry(void* param)
{
pthread_once(&key_once, make_key);

if (pthread_getspecific(key) == NULL)
{
pthread_setspecific(key, (void*)pthread_self());
}

printf("data=%u/n", pthread_getspecific(key));

return NULL;
}

pthread_once保证make_key只被执行一次。

方法二: 使用编译器扩展

示例:

__thread int i;

后者使用起来方便很多,但前者的好处是移植比较方便,即使编译器不支持,也可以自己封装相应的函数。


http://www.niftyadmin.cn/n/1229829.html

相关文章

十三、企业开发(5)

本章概要 数据校验 13.5 数据校验 数据校验时开发过程中一个常见的环节&#xff0c;一般来说&#xff0c;为了提高系统运行效率&#xff0c;都会在前端进行数据校验&#xff0c;但是这并不意味着不必在后端做数据校验了&#xff0c;因为用户还是可能在获取数据接口后手动传入…

系统程序员成长计划-内存管理(三)

转载时请注明出处和作者联系方式 文章出处&#xff1a;http://www.limodev.cn/blog 作者联系方式&#xff1a;李先静 <xianjimli at hotmail dot com> 内存管理器 在前面学习共享内存的时候&#xff0c;我们重新实现了循环队列&#xff0c;两个实现的不同之处只是在于内…

十四、应用监控(1)

本章概要 监控端点配置&#xff08;开启端点&#xff0c;暴露端点&#xff0c;端点保护&#xff0c;端点响应缓存&#xff0c;路径映射&#xff0c;CORS支持&#xff0c;健康信息&#xff0c;应用信息&#xff09; 当一个Spring Boot 项目运行时&#xff0c;开发者需要对 Spr…

十四、应用监控(2)

本章概要 监控信息可视化邮件报警 14.2 监控信息可视化 Spring Boot 中提供了监管信息管理段&#xff0c;用来实现监控信息的可视化&#xff0c;这样可以方便开发者快速查看系统运行情况&#xff0c;而不用一个一个地调用接口。 创建 Spring Boot Web 工程&#xff0c;添加以…

系统程序员成长计划-内存管理(四)

系统程序员成长计划&#xff0d;内存管理(四)转载时请注明出处和作者联系方式 文章出处&#xff1a;http://www.limodev.cn/blog 作者联系方式&#xff1a;李先静 <xianjimli at hotmail dot com> 惯用手法 《POSA(面向模式的软件架构)》中根据模式粒度把模式分为三类…

十五、项目构建与部署

本章概要 构建 JAR构建 WAR Spring Boot 项目可以内嵌 Servlet 容器&#xff0c;因此它的部署变得极为方便&#xff0c;可以直接打成可执行 JAR 包部署在有 Java 运行环境的服务器上&#xff0c;也可以像传统的 Java Web 应用程序那样打成 WAR 包运行。 15.1 JAR 15.1.1 项…

系统程序员成长计划-像机器一样思考(一)

系统程序员成长计划&#xff0d;像机器一样思考(一)Sunday, May 10th, 2009|Author: admin 转载时请注明出处和作者联系方式 文章出处&#xff1a;http://www.limodev.cn/blog 作者联系方式&#xff1a;李先静 <xianjimli at hotmail dot com> 变参函数的实现原理 C语言要…

一、Vue.js 概述

本章概要 Vue 2.x 响应式系统的实现原理Vue 3.0 响应式系统的实现原理体验 Vue 3.0 响应式系统Vue 3.0 带来的新变化 1.1 Web 前端技术的发展 略&#xff0c;自行百度。 1.2 MV* 模式 MVC 是 Web 开发中应用非常广泛的一种框架模式&#xff0c;之后又演变出 MVP 模式和 MV…