当前位置: 首页 > article >正文

线程控制与线程库

目录

解析tid

线程的地址空间布局

线程栈


我们来学习线程控制与线程库

解析tid

#include<iostream>
#include<string>
#include<cstdio>
#include<cstring>
#include<unistd.h>
#include<thread>
using namespace std;int shared_value = 100;
string toHex(pthread_t tid)
{char buffer[64];snprintf(buffer, sizeof(buffer), "0x%lx", tid);return buffer;
}void* start(void* args)
{string name = static_cast<const char*>(args);sleep(1);while (true){cout << "I am a new thread, name: " << name << ", thread is: " << toHex(pthread_self()) << ", shared_value: " << shared_value << ", &shared_value: " << &shared_value << endl;sleep(1);}return nullptr;
}int main()
{pthread_t tid;pthread_create(&tid, nullptr, start, (void*)"thread-1");cout << "I am a new thread, name: main, " << toHex(pthread_self())<< ", NEW thread id: " << toHex(tid) << endl;while (true){cout << "main thread, " << " shared_value: " << shared_value << ", &shared_value: " << &shared_value << endl;shared_value+=10;sleep(1);}pthread_join(tid, nullptr);return 0; 
}

通过上面这个代码的执行结果我们发现,两个线程都可以访问同一个全局变量的空间,重点不是这个,他们的id通过16进制打印出来有点像地址呀,没错线程的id不是一个单纯的数字串,而是线程属性集合在用户级线程库中的地址

果然是地址,怎么证明呢?,线程属性的地址具有唯一性对吧,线程的tid也具有唯一性,所以二者的具有相同的特点。

线程的地址空间布局

pthread_t 到底是什么类型呢?取决于实现。对于Linux⽬前实现的NPTL实现⽽⾔,pthread_t类 型的线程ID,本质就是⼀个进程地址空间上的⼀个地址。

对于Linux来说,其内核只有lwp,用户想要创建线程就需要调用用软件层封装的线程接口,那如果我们要得到线程的其他属性比如id,优先级,状态,栈大小,用户要调用线程库pthread.so就需要先将在用户内存的pthread.so线程库的地址通过页表映射到虚拟地址的共享区,在内存中线程库的内部维护着所有线程的属性集合,哦线程的属性存在这里,也就是线程的属性不是操作系统管理的,是线程库自己管理的,库管理线程属性集合依照先描述再组织的原则,对于比如线程大小,id状态什么的就装在一个struct结构体里面,struct pthread 是 线程控制块(Thread Control Block, TCB),用于存储线程的所有运行时信息。每个线程有独立的存储空间和线程栈,所以线程的属性=struct pthread(线程控制块)+线程局部存储+线程栈,这三个和起来就是线程的全部属性,这个线程栈是被动态申请出来的。tcb中维护描述库的控制结构

一个一个线程属性集合会依次排在库里面,此时库就相当于数组了,每个线程属性集合按照数组存放,此时线程id就是每个线程属性集合在线程库中的首地址,库里就这么设计的。

__thread(或 C11 的 _Thread_local)是 GCC/Clang 提供的 线程局部存储(Thread-Local Storage, TLS) 扩展,用于定义线程私有变量。每个线程拥有该变量的独立副本,互不干扰。__thread只能修饰内置类型,经由__thread修饰的变量相当于将这个变量在每个线程属性集的局部存储中各自拷贝一份。相当于__thread修饰的变量在各个线程中相互独立了。

__thread int shared_value = 100;string toHex(pthread_t tid)
{char buffer[64];snprintf(buffer, sizeof(buffer), "0x%lx", tid);return buffer;
}

我们可以看到不仅子线程中的全局变量不再随着主线程++了,就连两个线程的同一个名字的全局变量的地址都不一样了。

之前有一个mmap区域,mmap 在 Linux 线程(尤其是 pthread 线程)的实现中扮演重要角色,但 不直接创建线程,而是用于 分配线程栈和线程局部存储(TLS),mmap创建线程的物理空间,mmap是你们malloc的底层实现,mmap首先会先向物理内存申请一块空间,然后通过页表映射到库里面,建立完映射,然后创建一个虚拟地址空间让虚拟内存中的一个指针指向这个物理内存空间,然后允许用户使用的部分返回。这个我们之前讲过,是靠着缺页中断完成最后的合法完整映射的。

线程的tcb属性控制结构体在调用用户级创建线程函数pthread_create就已经创建好了,对于clone函数操作系统内核创建线程通过syscall/int 80中断时CPU陷入内核执行中断方法促使操作系统去创建线程。在创建的过程中线程的属性是会动态变化的,在上层可以查看。

最后结论:每一个线程都有自己的栈结构,新线程的栈结构是在共享区重新申请空间,然后将起始地址写入tcb中的stack指针里面(所以指向线程栈的地址在tcb里面)最后通过clone函数传进内核。

线程栈

虽然Linux将线程和进程不加区分的统⼀到了task_struct ,但是对待其地址空间的 stack 还是 有些区别的。

这个线程栈的大小是固定的,size存在线程的tcb里面,对于Linux进程或者说主线程,简单理解就是main函数的栈空间,在fork的时候,实际上就是复 制了⽗亲的 stack 空间地址,然后写时拷⻉(cow)以及动态增⻓。如果扩充超出该上限则栈溢出 会报段错误(发送段错误信号给该进程)。进程栈是唯⼀可以访问未映射⻚⽽不⼀定会发⽣段错 误⸺⸺超出扩充上限才报。

然⽽对于主线程⽣成的⼦线程⽽⾔,其 stack 将不再是向下⽣⻓的,⽽是事先固定下来的。线 程栈⼀般是调⽤glibc/uclibc等的 pthread 库接⼝ pthread_create 创建的线程,在⽂件映 射区(或称之为共享区)。

这种stack不能动态增⻓,⼀旦⽤尽就没了,这是和 ⽣成进程的fork不同的地⽅。因此,对于⼦线程的 stack ,它其实是在进程的地址空间中mmap出来的⼀块内存区域,原则上是 线程私有的,但是同⼀个进程的所有线程⽣成的时候,是会浅拷⻉⽣成者的 task_struct 的很多 字段,如果愿意,其它线程也还是可以访问到的,于是⼀定要注意。

根据clone的参数,调用clone的时候,clone会使用 mmap 分配栈的大小和空间。

相关文章:

线程控制与线程库

目录 解析tid 线程的地址空间布局 线程栈 我们来学习线程控制与线程库 解析tid #include<iostream> #include<string> #include<cstdio> #include<cstring> #include<unistd.h> #include<thread> using namespace std;int shared_val…...

P1182 数列分段 Section II

P1182 数列分段 Section II - 洛谷 题目描述 对于给定的一个长度为 N 的正整数数列 A1​∼AN​&#xff0c;现要将其分成 M&#xff08;M≤N&#xff09;段&#xff0c;并要求每段连续&#xff0c;且每段和的最大值最小。 关于最大值最小&#xff1a; 例如一数列 4 2 4 5 1…...

比手动备份快 Iperius全自动加密备份,NAS/云盘/磁带机全兼容

IperiusBackupFull是一款专为服务器和工作站设计的备份解决方案&#xff0c;它同时也是一款针对Windows 7/8/10/11/Server系统的简洁且可靠的备份软件。该软件支持增量备份、数据同步以及驱动器镜像&#xff0c;确保能够实现完全的系统恢复。在备份存储方面&#xff0c;Iperius…...

从概率到梯度:理解分类问题中交叉熵的优越性

分类问题一般使用交叉熵&#xff08;Cross-Entropy&#xff09;而不是平方损失&#xff08;Square Loss&#xff09;函数1. **概率解释**2. **梯度性质**3. **对错误的惩罚**4. **计算复杂度**5. **总结** 分类问题一般使用交叉熵&#xff08;Cross-Entropy&#xff09;而不是平…...

2025最新版Ubuntu Server版本Ubuntu 24.04.2 LTS下载与安装-详细教程,细致到每一步都有说明

官网 https://ubuntu.com/ 下载 点击菜单 Prodercts> Ubuntu OS>Ubuntu Server 点击下载 下载后会有个弹窗 安装 选择第一个 install Ubuntu Server 直接默认&#xff0c;选择English 【默认】 选择键盘布局【默认】 选择安装配置【默认】 配置网络 我这里选择…...

更新测试环境构建命令以解决构建失败问题

本段代码解决 更新测试环境构建命令以解决构建失败问题 //本项目是reactumi3antdesign 搭建的后台管理系统 "build:test": "cross-env UMI_ENVtest NODE_OPTIONS--openssl-legacy-provider umi build"**原因&#xff1a;**Node.js v17 的 OpenSSL 3.0 与旧…...

C 语言中, scanf 函数在哪些情况下会结束输入读取:

在 C 语言中&#xff0c; scanf 函数在以下几种情况下会结束输入读取&#xff1a; &#xff1a; 1. 遇到指定格式匹配失败&#xff1a; scanf 按照格式字符串要求读取输入。当输入数据格式与格式字符串不匹配时&#xff0c;就会结束读取。例如 scanf(“%d”, &num) 要求输…...

树莓派5-GPIO和40针引脚

1.树莓派5引脚图 2.GPIO 引脚作用 (1) 电压 板上有两个 5V 引脚和两个 3.3V 引脚&#xff0c;以及一些不可配置的接地引脚 (0V)。其余引脚均为通用 3.3V 引 脚&#xff0c;这意味着输出设置为 3.3V&#xff0c;输入可接 3.3V。 (2) 输出 指定为输出引脚的 GPIO 引脚可设置为…...

【数据库】sql错题详解

1. 执行子查询 SELECT 供应商号 FROM 订购单 WHERE 职工号 IN (E1, E3) GROUP BY 供应商号 HAVING COUNT(DISTINCT 职工号) 2筛选职工号为 E1 或 E3 的记录&#xff1a; 依据 WHERE 职工号 IN (E1, E3) 这个条件&#xff0c;从 订购单 表中把职工号为 E1 或者 E3 的记录筛选出…...

搭建主从DNS、nfs、nginx

任务需求&#xff1a; 客户端通过访问 www.nihao.com 后&#xff0c;能够通过 dns 域名解析&#xff0c;访问到 nginx 服务中由 nfs 共享的首页文件&#xff0c;内容为&#xff1a;Very good, you have successfully set up the system. 各个主机能够实现时间同步&#xff0c;…...

C#重写treeView控件

1.先准备两张图片downdrop.png、downdrop_open.png放在项目Resources里 2.新建用户控件BaseTreeView控件 3.重写控件继承TreeView&#xff0c;记得删除AutoScaleMode这一行&#xff0c;否则会报错 public partial class BaseTreeView : TreeView {//这个属性貌似不起作用&…...

JS自动装箱(Auto-boxing)机制深度解析

&#x1f4e6; JS自动装箱(Auto-boxing)机制深度解析 自动装箱&#xff08;Autoboxing&#xff09; 是 JavaScript 的一项特性 &#x1f31f; 核心概念速览 自动装箱 原始值临时变身对象 当对原始值调用方法或访问属性时&#xff0c;JS 引擎会自动将其转换为对应的包装对象&…...

ArcGIS 10.8.1之后发布栅格数据的MapServer 动态工作空间 替换数据源渲染问题

背景 经过测试&#xff0c;Server 10.8.1、11.0、11.1发布相关服务设置动态空间之后&#xff0c;前端都无法自动读取同名的clr色彩映射表文件进行渲染&#xff0c;服务都是由ArcGIS Pro进行发布。 原因 基于ArcMap发布的服务才支持&#xff0c;但是10.8.1之后不支持ArcMap发…...

Java集合框架深度剖析:从数据结构到实战应用

引言 Java集合框架是Java开发中的核心组件之一&#xff0c;其设计目标是提供高性能、高复用性的数据容器。无论是数据处理、缓存设计还是高并发场景&#xff0c;集合框架都扮演着关键角色。本文将从List、Map、Set三大核心接口出发&#xff0c;深入剖析其主流实现类&#xff0…...

【MySQL】监控MySQL

目录 使用状态变量监控MySQL 使用性能模式&#xff08;Performance Schema&#xff09;监控MySQL 1.性能模式 2.性能模式设置表 3.sys模式 使用状态变量监控MySQL 使用 show status 语句评估系统运行状况。 可以添加范围修饰符global或session来显示全局或本地状态信息。…...

涅槃上岸,入陕进军,复试全程流程开启!

复试决胜局&#xff0c;整装待发&#xff0c;上岸西电&#xff01; 线下复试注意事项、全流程、录取后西安旅游提前告知&#xff01; 过两天考研复试笔试、机试&#xff08;如果有&#xff09;、面试就要开始了&#xff0c;我们需要准备很多东西&#xff0c;学长从以下几个方面…...

msyql--基本操作之运维篇

检查 root 用户的权限 查看该用户针对这个数据库的权限 -- 如果在终端连接mysql时需要 mysql -u root -p -- 查看用户权限 SELECT user, host FROM mysql.user WHERE user root;可以看的出来root有他的访问权限&#xff0c;如过没有localhost或者% 说明没有访问权限 添加…...

鸿蒙开发:openCustomDialog关闭指定Dialog

前言 本文基于Api13 openCustomDialog弥补了CustomDialogController在使用上存在的诸多限制&#xff0c;实现了可以在任意位置上弹出&#xff0c;可以说是非常的方便&#xff1b;但是&#xff0c;在使用的时候遇到了一些小阻碍&#xff0c;比如一个页面中可能存在多个弹窗&…...

es6 fetch

对比XHR &#x1f6e0;️ fetch 所有配置项 fetch(url, {// 核心配置 method: GET, // HTTP 方法: GET, POST, PUT, DELETE, PATCH, HEAD, OPTIONSheaders: { // 请求头&#xff08;支持 Headers 对象或普通对象&#xff09;Content-Type: applicati…...

Apache Tomcat RCE漏洞(CVE-2025-24813)

一&#xff0c;漏洞描述 该漏洞在于 Tomcat 在处理不完整PUT请求上传时&#xff0c;会使用了一个基于用户提供的文件名和路径生成的临时文件。 二&#xff0c;漏洞条件 1&#xff0c;默认 Servlet 启用了写权限&#xff08;默认禁用&#xff09; 2&#xff0c;启用了部分PUT…...

Apache HttpClient使用

一、Apache HttpClient 基础版 HttpClients 是 Apache HttpClient 库中的一个工具类&#xff0c;用于创建和管理 HTTP 客户端实例。Apache HttpClient 是一个强大的 Java HTTP 客户端库&#xff0c;用于发送 HTTP 请求并处理 HTTP 响应。HttpClients 提供了多种方法来创建和配…...

智能汽车图像及视频处理方案,支持视频星轨拍摄能力

美摄科技作为智能汽车图像及视频处理领域的先行者&#xff0c;正以革新性的技术引领着行业的未来发展。美摄科技智能汽车图像及视频处理方案&#xff0c;一个集高效性、智能化、画质增强于一体的创新解决方案&#xff0c;旨在重塑智能汽车图像画质的新标准&#xff0c;并支持前…...

【微服务架构】本地负载均衡的实现(基于随机算法)

前言 负载均衡 概念&#xff1a;一种将网络流量或业务请求均匀分配到多个服务器或服务实例上的技术&#xff0c;旨在提高系统的可用性、性能和可伸缩性。作用&#xff1a; 提高性能&#xff1a;通过将请求分散到多个实例上&#xff0c;避免单个实例因请求过多而过载&#xff…...

C盘急救实录:从爆红到畅快

极速救援通道&#xff08;懒人专享&#xff09; 老规矩&#xff0c;先上王炸方案&#xff01;”小番茄C盘清理器”直达链接&#xff1a;https://cclean-cdn.xkbrowser.com/cleanmaster/FanQieClean_13046_st.exe 这个神器有三绝&#xff1a; 智能扫描引擎&#xff1a;能识别23…...

从零开始理解基于深度学习的语义分割模型:RCA与RCM模块的实现

从零开始理解基于深度学习的语义分割模型:RCA与RCM模块的实现 随着深度学习技术的发展,图像分割任务取得了长足的进步。本文将从一个具体的PyTorch代码实例出发,带大家了解一种 novel 的语义分割网络架构——RCA(Rectangular Self-Calibration Attention)和 RCM(Rectang…...

openGl片段着色器的含义

片段着色器的含义及代码中的应用说明&#xff1a; 1. 片段着色器的基本概念 片段着色器&#xff08;Fragment Shader&#xff09;是OpenGL着色器管线中的关键组件&#xff0c;主要用于计算屏幕空间中每个片段&#xff08;对应像素&#xff09;的最终颜色。它是图形渲染流程的…...

ROS2 部署大语言模型节点

4GB GPU的DeepSeek-Coder 1.3B模型&#xff0c;并且它已经被量化或优化过。以下是具体的步骤&#xff1a; 安装必要的依赖项&#xff1a; pip install transformers torch grpcio googleapis-common-protos创建一个新的ROS 2包&#xff1a; cd ~/ros2_ws/src ros2 pkg creat…...

UART转APB模块ModelSim仿真

一、简介 之前介绍过一个UART转AHB模块&#xff0c;这个代码的框架有个好处&#xff0c;就是FPGA内总线接口比较容易修改成其他总线接口。下图是UART转AHB模块中子模块uart_ahb_mst的框图&#xff0c;主要有三个状态机&#xff1a; &#xff08;1&#xff09; UART_RX_FSM将接收…...

【LeetCode 题解】算法:4.寻找两个正序数组的中位数

1. 引言&#xff1a;挑战 LeetCode 经典算法题 在算法这片广袤无垠的天地里&#xff0c;一道道经典题目宛如夜空中熠熠生辉的星辰&#xff0c;持续吸引着开发者们投身其中&#xff0c;不断探索。今天&#xff0c;我们继续将目光聚焦于 LeetCode 平台上一道极具代表性的题目&am…...

基于 SGLang 部署 Qwen2.5 7B 模型

本文将详细介绍如何使用 SGLang 快速部署 Qwen2.5 7B 模型,并深入探讨 SGLang 的关键性能优化技术,以及预期可以达到的延迟和吞吐量。 1. SGLang 框架介绍 SGLang 旨在解决 LLM 服务中的核心挑战: 高延迟: LLM 推理通常需要较长的计算时间,导致响应延迟高。低吞吐量: 由…...