惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

酷 壳 – CoolShell
酷 壳 – CoolShell
H
Hacker News: Front Page
P
Palo Alto Networks Blog
T
ThreatConnect
Apple Machine Learning Research
Apple Machine Learning Research
博客园_首页
T
True Tiger Recordings
P
Privacy & Cybersecurity Law Blog
B
Blog
IT之家
IT之家
Last Week in AI
Last Week in AI
F
Full Disclosure
Hacker News: Ask HN
Hacker News: Ask HN
C
Comments on: Blog
Microsoft Azure Blog
Microsoft Azure Blog
C
Cybersecurity and Infrastructure Security Agency CISA
Microsoft Security Blog
Microsoft Security Blog
博客园 - 【当耐特】
N
News and Events Feed by Topic
NISL@THU
NISL@THU
腾讯CDC
雷峰网
雷峰网
Security Latest
Security Latest
李成银的技术随笔
M
Microsoft Research Blog - Microsoft Research
L
LangChain Blog
L
Lohrmann on Cybersecurity
cs.CL updates on arXiv.org
cs.CL updates on arXiv.org
C
Check Point Blog
Y
Y Combinator Blog
Recent Announcements
Recent Announcements
博客园 - Franky
N
News | PayPal Newsroom
V
V2EX
A
About on SuperTechFans
The Register - Security
The Register - Security
月光博客
月光博客
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
Google Online Security Blog
Google Online Security Blog
MyScale Blog
MyScale Blog
Cisco Talos Blog
Cisco Talos Blog
Vercel News
Vercel News
WordPress大学
WordPress大学
C
Cyber Attacks, Cyber Crime and Cyber Security
The Hacker News
The Hacker News
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
爱范儿
爱范儿
A
Arctic Wolf
L
LINUX DO - 最新话题
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More

博客园 - PKICA

汇编语言语法详解 gdb汇编调试 gdb-pwndbg的安装与使用指南 gdb调试插件gef linux系统readelf命令使用指南 gcore转储进程内存 gdb查看命令 RGB与YUV颜色编码的区别 Rust原子类型 C++ STL求两个集合交集差集 gdb调试集锦 ubuntu24.0.4使用root用户登录 ubuntu24.0.4输入密码后跳回登录界面 AI内存压缩技术TurboQuant及存疑 ubuntu切换到指定内核版本 在没有顶级科技大佬直接背书的情况下deepseek为啥能够异军突起? HuggingFace和deepseek的关系 当前主流AI大模型 Rust写时克隆Cow系列2
C语言thread_local
PKICA · 2026-04-14 · via 博客园 - PKICA

C语言中的 thread_local(在C11标准中为 _Thread_local,通常建议使用 <threads.h> 中的 thread_local 宏)用于声明线程局部存储 (Thread-Local Storage, TLS) 变量。

这种变量在每个线程中都有一个独立的实例,线程之间互不干扰,即使它们共享同一个变量名,在多线程环境下使用它不仅是线程安全的,还避免了互斥锁带来的性能开销。

1. thread_local 的核心特性

  • 线程生存期: 变量在线程启动时生成,在线程结束时销毁。
  • 独立副本: 每个线程存取的是属于自己的副本。
  • 初始化: 不同于普通变量,线程局部变量的值在线程启动时被初始化。
  • 修饰限制: 只能用于全局变量、文件范围的静态变量或函数内部的静态变量。

2. 使用方法 (C11 标准)

C11 标准引入了 _Thread_local 关键字。通常引入 <threads.h> (如果编译器支持) 或在 C23 标准中直接使用 thread_local。 

#include <stdio.h>
#include <threads.h> // C11线程标准库

// 定义一个线程局部变量
thread_local int thread_specific_var = 0;

int main_thread_func(void* arg) {
    thread_specific_var = 10; // 当前线程修改
    printf("Thread %d: var = %d\n", (int)thrd_current(), thread_specific_var);
    return 0;
}

int main() {
    thrd_t t1, t2;
    // 启动两个线程
    thrd_create(&t1, main_thread_func, NULL);
    thrd_create(&t2, main_thread_func, NULL);
    
    // 等待线程结束
    thrd_join(t1, NULL);
    thrd_join(t2, NULL);

    return 0;
}

在上面的示例中,如果 thread_local 没有被定义,两个线程会共享 thread_specific_var 并产生冲突。使用了 thread_local 后,每个线程会打印出各自的 10,互不干扰。

3. 不同标准下的写法

  • C11 / C17: 使用 _Thread_local#include <threads.h> 后的 thread_local
  • C23 : 直接使用 thread_local
  • GCC/Clang 扩展: 使用 __thread 关键字(static __thread int var;),它更早期且实现效率较高。

4. 关键注意事项

  • 静态/全局修饰: 如果在块作用域(函数内部)使用,必须加上 staticextern
  • 性能开销: 线程局部变量的读取通常比全局变量稍慢,且随着线程数量增加,内存占用会增大。
  • 复杂析构函数: 如果线程局部变量需要复杂的析构函数(在C++中更常见),可能会在线程退出时导致意外的释放后使用(use-after-free)问题。
  • 与POSIX pthread_key 区别: thread_local 是一种语言层面的特性,使用起来更简单、更高效,不再需要像 pthread_key_create 那样处理 void* 指针和显式管理内存。

一般,“标准方式”和“编译器扩展方式”区别:

定义标准 C11 标准关键字 GCC/Clang 编译器扩展 引入方式 包含 <threads.h> 后可用 thread_local 宏 原生支持,无需头文件 约束限制 仅能用于全局或 static 变量 同左 初始化 只能用常量表达式初始化 只能用常量表达式初始化性能 理论一致 在老版本 GCC 上可能略快
特性_Thread_local (C11)__thread (GNU/Clang)

总结

在多线程程序中,如果需要维护线程特有的状态(如错误码 errno、特定线程的日志缓存等),thread_local 是现代C语言(C11及以后)推荐的线程安全实现方式。

参考资料: