【Linux】深入理解线程

news2025/1/18 14:00:33

在这里插入图片描述

👦个人主页:Weraphael
✍🏻作者简介:目前正在学习c++和算法
✈️专栏:Linux
🐋 希望大家多多支持,咱一起进步!😁
如果文章有啥瑕疵,希望大佬指点一二
如果文章对你有帮助的话
欢迎 评论💬 点赞👍🏻 收藏 📂 加关注😍


目录

  • 一、线程ID
  • 二、线程独立栈
  • 三、线程局部存储
  • 四、代码

一、线程ID

pthread_self函数是POSIX线程库(pthread)提供的一个函数,用于获取当前线程的线程ID

【函数原型】

#include <pthread>
pthread_t pthread_self(void);

说明:

  • 功能:pthread_self函数返回调用线程的线程ID,类型为pthread_t。线程ID是一个唯一标识符,用于区分不同的线程。
  • 返回值:返回当前线程的线程ID
#include <iostream>
#include <pthread.h>
#include <unistd.h>
#include <string>
 
using namespace std;

void *thread_task(void *)
{
    while (true)
    {
        cout << "new thread say: My id is " << pthread_self() << endl;
        sleep(1);
    }
    return nullptr;
}

int main()
{
    pthread_t tid;
    pthread_create(&tid, nullptr, thread_task, nullptr);

    cout << "main thread say: new thread id is " << tid << endl;

    pthread_join(tid, nullptr);

    return 0;
}

【程序结果】

在这里插入图片描述

奇怪?打印出来的pthread_t对象的值怎么和ps -aL命令显示的LWP值不一样?

打印出来默认是一个十进制数字,如果用十六进制打印又会是什么结果呢?

#include <iostream>
#include <pthread.h>  
#include <unistd.h>
#include <string>
 
using namespace std;

void *thread_task(void *)
{
    while (true)
    {
        printf("new thread say: My id is %x\n", pthread_self());
        sleep(1);
    }
    return nullptr;
}

int main()
{
    pthread_t tid;
    pthread_create(&tid, nullptr, thread_task, nullptr);

    printf("main thread say: new thread id is %x\n", tid);

    pthread_join(tid, nullptr);

    return 0;
}

【程序结果】

在这里插入图片描述

我们发现:打印结果很像地址。所以这个pthread_t对象到底是什么东西呢?

【正文开始】

我们知道,Linux操作系统没有真正意义上的线程(用的是进程的数据结构来模拟的),只有轻量级进程的概念,所以操作系统不会直接提供线程的系统调用,只会提供轻量级进程的系统调用

具体来说,Linux中的轻量级进程通过clone()系统调用来创建clone()系统调用允许创建一个新的进程,可以共享某些资源(如内存空间),同时也可以选择不共享其他资源(如文件描述符),从而实现了轻量级进程的创建。

在这里插入图片描述

但是我们一般不会使用轻量级进程的系统调用,你看看上面的接口参数就知道了,clone()系统调用的参数非常复杂。所以,为了更方便地使用轻量级进程(即线程),开发者们开发了线程库(pthread,来提供更易用的接口。该库底层封装了clone()调用,隐藏了复杂的系统级细节,使得另一批开发者可以很简单的使用线程了。

但你不得会发现几个细节:

  1. 第一个参数int (*fn)(void *):是一个函数指针。指向新进程(或轻量级进程)所要执行的函数,和pthread_create()的第三个参数是一样的。
  2. 第二个参数void *child_stack:是新线程的栈空间。通常需要在调用clone()之前先为新进程分配一段栈空间,并将这个栈的起始地址传递给child_stack参数。
  3. 其他参数不关心。

因为pthread库在Linux中是一个动态库,当我们调用此库的API时,操作系统要把它从磁盘加载到内存,然后通过页表映射到进程的地址空间的共享区中。此时该进程内的所有线程都是能看到这个动态库

在这里插入图片描述

我们说每个线程都有自己私有的栈,其中主线程采用的栈是进程地址空间中原生的栈,而其余线程采用的栈就是在pthread库中开辟的,也就是在共享区中开辟的。而用户可能会创建多个线程,那么pthread库中就有多个回调函数、私有栈、线程标识符等属性。因此,为了区分这些属性属于哪个线程,pthread库也要对这些线程进行管理,即先描述再组织

所以,除了主线程以外,我们每创建一个线程,在线程库当中就要为该线程创建用户层的线程控制块TCB(包含线程的属性)。而这种由用户空间的线程库(pthread库)管理和调度的线程我们称之为用户级线程

回过头来,因此我们要找到一个用户级线程只需要找到该线程在进程地址空间中共享区的起始地址,然后就可以获取到该线程的各种信息。 所以说,pthread_t对象本质就是进程地址空间的共享区上的一个虚拟地址。因为同一个进程中所有的虚拟地址都是不同的,因此可以用它来唯一区分每一个线程。

另外,ps -aL命令显示的LWP值可以理解为:LWPLinux内核管理轻量级进程的标识符。因为不管是进程还是线程,都由相同的数据结构task_struct表示;又因为线程是轻量级进程,是操作系统调度的基本单位。因此,LWP值属于内核进程调度的范畴,确保线程能够有效地被调度

总结:

  • Linux线程的概念是由pthread库来维护和实现的。即Linux的线程 = 用户级线程 + 内核的LWP
  • 线程可以分为用户级线程和内核级线程,而Linux就属于用户级线程。

在这里插入图片描述

二、线程独立栈

线程之间存在独立的栈结构,并且栈是由库给我们提供的(上面说过),这可以保证彼此之间执行任务时不会相互干扰

验证方式:创建5个线程调用同一个函数,并打印该函数中的临时变量的地址。

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

using namespace std;

void *thread_task(void *)
{
    int stack_tmp = 1;
    printf("线程id: 0x%x, 变量stack_tmp的地址: %p\n", pthread_self(), &stack_tmp);
    return nullptr;
}

int main()
{
    vector<pthread_t> tids;

    for (int i = 1; i <= 5; i++)
    {
        pthread_t tid;
        /*
          注:不要给pthread_create的第四个参数传对传for循环中的对象。
			因为当第一次for循环结束后,for循环内的对象销毁,
			那么第一个线程参数的指针就会指向已经销毁的对象,造成野指针!
			
			如果真的想要在传循环中的对象,可以现在堆上创建对象     
        */
        pthread_create(&tid, nullptr, thread_task, nullptr);
        tids.push_back(tid);
        sleep(1);
    }

    for (int i = 0; i < tids.size(); i++)
    {
        pthread_join(tids[i], nullptr);
    }

    return 0;
}

【程序结果】

在这里插入图片描述

可以看到5个线程调用同一个函数threadRun,打印的 “同一个” 变量stack_tmp的地址都是不相同的,说明线程在调用threadRun函数使用的是自身的栈结构,这足以证明线程独立栈的存在。

但如果我们主线程就想要访问上面任意一个线程的变量,可以做到吗?

当然可以做到,因为主线程和其他线程共享地址空间,其他线程的栈是由共享区中的库为线程创建的,因此主线程一定是能做到的。

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

using namespace std;

int *p = nullptr; // 获取第一个线程中的变量stack_tmp

string toHex(pthread_t t)
{
    char buffer[64];
    snprintf(buffer, sizeof(buffer), "0x%x", t);
    return buffer;
}

struct threadData
{
    string threadname;
};

void InitThreadData(threadData *td, int number)
{
    td->threadname = "thread-" + to_string(number);
}

void *thread_task(void *args)
{
    threadData *td = (threadData *)args;

    int stack_tmp = 1;
    if (td->threadname == "thread-2")
    {
        p = &stack_tmp;
    }

    int cnt = 3;
    while (cnt--)
    {
        cout << "线程名字:" << td->threadname
             << ", 变量stack_tmp的地址: " << &stack_tmp
             << ", stack_tmp = " << stack_tmp << endl;
        sleep(1);
    }

    return nullptr;
}

int main()
{
    vector<pthread_t> tids;
    for (int i = 1; i <= 3; i++)
    {
        pthread_t tid;

        threadData *td = new threadData;
        InitThreadData(td, i);

        pthread_create(&tid, nullptr, thread_task, (void *)td);

        tids.push_back(tid);
        sleep(1);
    }

    sleep(1);
    cout << "main thread get a thread_two local value, val: " << *p 
    	 << ", &val: " << p << endl;

    for (int i = 0; i < tids.size(); i++)
    {
        pthread_join(tids[i], nullptr);
    }

    return 0;
}

【程序结果】

在这里插入图片描述

线程和线程当中没有秘密,只不过我们要求每一个线程有自己独立的栈,但是线程是共享地址空间的,线程的栈上的数据,也是可以被其他线程看到并且访问的。如果我们一个线程想要访问另一个线程的值,当然可以访问,只不过我们平时禁止这样做。因此我们说线程之间的栈是独立的,而不是私有的!

总结

  • 所有线程都要有自己独立的栈结构(独立栈),其中主线程采用的栈是进程地址空间中原生的栈,而其余线程采用的栈就是在pthread库中开辟的。
  • 线程和线程当中没有秘密。

三、线程局部存储

线程之间共享全局变量,对全局变量 进行操作时,会影响其他线程

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

using namespace std;

int cnt = 1;

void *thread_task(void *)
{
    printf("thread: 0x%x, cnt = %d, &cnt = %p\n", pthread_self(), cnt, &cnt);
    ++cnt;
    return nullptr;
}

int main()
{
    vector<pthread_t> tids;
    for (int i = 1; i <= 3; i++)
    {
        pthread_t tid;

        pthread_create(&tid, nullptr, thread_task, nullptr);

        tids.push_back(tid);
        sleep(1);
    }

    for (int i = 0; i < tids.size(); i++)
    {
        pthread_join(tids[i], nullptr);
    }

    return 0;
}

【程序结果】

在这里插入图片描述

从以上结果看出,三个线程使用的全局变量cnt是一个共享资源,即全局变量是被所有的线程看到并同时访问的。

如何让全局变量私有化呢?即每个线程看到的全局变量不同。

可以用 __thread修饰,修饰之后,全局变量不再存储至全局数据区,而且存储至线程的局部存储区中。注意:__thread是编译器提供的特性,只能定义内置类型,不能用来修饰自定义类型

__thread int cnt = 1;

在这里插入图片描述

如上,修饰之后,每个线程确实看到了不同的 “全局变量”。但此时的 “全局变量” 的地址变大了

这是因为__thread修饰后的变量不再存储在全局数据区 中,而是存储在线程的局部存储区中。线程的局部存储区位于 共享区,而共享区(处于堆栈之间)的地址是天然大于全局数据区的

在这里插入图片描述

__thread修饰变量有什么好处呢?

  • 线程安全性:每个线程都有自己独立的变量实例,不同线程之间互不干扰。这种方式避免了多线程并发访问全局变量时可能出现的竞态条件和数据污染问题
  • 效率提高线程局部变量的访问速度通常比全局变量更快。因为线程局部变量存储在每个线程自己的内存空间中,线程可以直接访问自己的变量实例,而不需要加锁或者进行其他同步操作。

四、代码

Gitee链接:点击跳转

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1976997.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于ssm的图书管理系统/图书借阅管理系统

获取源码联系方式请查看文章结尾&#x1f345; 摘 要 网络技术的快速发展给各行各业带来了很大的突破&#xff0c;也给各行各业提供了一种新的管理模块&#xff0c;对于图书管理将是又一个传统管理到智能化信息管理的改革&#xff0c;对于传统的图书借阅的管理&#xff0c;所包…

最长上升子序列LIS(一般+优化)

1. 题目 题目链接&#xff1a; B3637 最长上升子序列 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 输入样例&#xff1a; 6 1 2 4 1 3 4 输出样例&#xff1a; 4 说明/提示&#xff1a; 分别取出 1、2、3、4 即可。 2. 具体实现 2.1 一般做法 dp[i]表示第i个位置的…

医院管理系统读取身份证源码- CyberWinApp-SAAS 本地化及未来之窗行业应用跨平台架构

一、身份证读取 提高效率&#xff1a;快速获取身份信息&#xff0c;避免手动输入的繁琐和耗时&#xff0c;极大地提升业务办理速度。 准确性高&#xff1a;减少人工输入错误&#xff0c;确保身份信息的精准无误。 便捷操作&#xff1a;简化流程&#xff0c;使工作人员操作更轻…

51单片机个人学习笔记14(直流电机驱动及PWM)

前言 本篇文章属于STC89C52单片机&#xff08;以下简称单片机&#xff09;的学习笔记&#xff0c;来源于B站教学视频。下面是这位up主的视频链接。本文为个人学习笔记&#xff0c;只能做参考&#xff0c;细节方面建议观看视频&#xff0c;肯定受益匪浅。 [1-1] 课程简介_哔哩…

Elemnt UI筛选时间功能

html&#xff1a; <el-form-item label"数据筛选: " ><el-date-picker v-model"choose_time" type"datetimerange" size"small" change"chooseTime" style"width:100%;" value-format"yyyy-MM-dd …

逻辑推理之lora微调

逻辑推理微调 比赛介绍准备内容lora微调lora微调介绍lora优势代码内容 start_vllm相关介绍调用 运行主函数提交结果总结相应连接 比赛介绍 本比赛旨在测试参与者的逻辑推理和问题解决能力。参与者将面对一系列复杂的逻辑谜题&#xff0c;涵盖多个领域的推理挑战。 比赛的连接:…

内网穿透--ICMP隧道转发实验

实验背景 通过公司带有防火墙功能的路由器接入互联网&#xff0c;然后由于私网IP的缘故&#xff0c;公网无法直接访问内部web服务器主机。通过内网其它主机做代理&#xff0c;穿透访问内网web服务器主机边界路由器或防火墙做静态NAT映射访问内网服务器inux主机&#xff0c;且策…

C++分析AVL树

目录 AVL树介绍 AVL树平衡因子更新分析 AVL树插入时旋转与平衡因子更新 左单旋 右单旋 左右单旋 右左单旋 AVL旋转可行性 AVL树节点删除&#xff08;待补充&#xff09; AVL树分析 AVL树介绍 二叉搜索树在某些极端情况下可能会退化&#xff0c;为了解决这个问题&…

Redis学习[6] ——Redis缓存设计

八、Redis缓存设计 8.1 为什么Redis用作缓存&#xff1f; 一般来说&#xff0c;数据库的数据都是落在磁盘上的&#xff0c;会导致读写速度很慢。如果用户的请求量非常大&#xff0c;数据库很容易崩溃。由于Redis的数据保存在内存中&#xff0c;读写速度很快&#xff0c;所以R…

SQL注入 报错注入+附加拓展知识,一篇文章带你轻松入门

第5关--------------------------------------------> 前端直接不会显示账号密码的打印&#xff1b;但是在接收前端的数据的那部分后端那里&#xff0c;会看前端传递过来的值是否正确&#xff0c;如果不正确&#xff0c;后端接收值那里就会当MySQL语句执行错误&#xff0c;…

RK3568笔记五十一:W25Q64测试(spi 标准接口 )

若该文为原创文章&#xff0c;转载请注明原文出处。 前面有测试过W25Q64&#xff0c;但那是自己编写的驱动&#xff0c;现在使用内核自带的驱动&#xff0c;只需要通过SPI标准接口&#xff0c;编写应用程序即可以读写W25Q64. 一、硬件原理图 SPI 引脚 功能 MOSI GPIO3_C1 …

【java基础】徒手写Hello, World!程序

文章目录 前提&#xff1a;java环境变量配置使用vscode编写helloworld解析 前提&#xff1a;java环境变量配置 https://blog.csdn.net/xzzteach/article/details/140869188 使用vscode编写helloworld code .为什么用code看下图 报错了&#xff01;&#xff01;&#xff01;&…

【MATLAB】Matlab安装包及验证生成器

通过百度网盘分享的文件&#xff1a;Matlab 链接: https://pan.baidu.com/s/1PF8iP31WFJUYRF7PLyiX2A?pwdxkds 提取码&#xff1a;xkds

简单搭建dns服务器

目录 一.安装服务 二.编写子配置文件 三.编写主配置文件 四.编写文件 五.测试 一.安装服务 [rootnode1 ~]# dnf install bind -y 二.编写子配置文件 [rootnode1 ~]# vim /etc/named.rfc1912.zones 三.编写主配置文件 [rootnode1 ~]# vim /etc/named.conf 四.编写文件 …

一款创新的物联网综合业务支撑平台,提供资费、客户、进销存、合同、订单、续费、充值、账单等功能(附源码)

前言 在当今快速发展的物联网时代&#xff0c;企业和开发者面临着很大的挑战和机遇。现有软件在处理物联网设备和数据管理方面常常存在一些痛点&#xff0c;如设备管理分散、数据同步不及时、用户交互体验不佳等。这些问题不仅影响了物联网解决方案的效率&#xff0c;也限制了…

docker部署可执行的jar

1.将项目打包&#xff0c;上传到服务器的指定目录 2.在该目录下创建Dockerfile文件 3.Dockerfile写入如下指令 # 基于哪个镜像 FROM java:8 # 拷贝文件到容器&#xff0c;也可以直接写成ADD xxxxx.jar /app.jar ADD springboot-file-0.0.1.jar file.jar RUN bash -c touch /…

GuLi商城-商品服务-API-新增商品-调试会员等级相关接口

在网关服务中配置路由: 代码: nacos这些服务都要启动: 如果有不是一个命名空间中的,要改成同一个命名空间中 启动商品product服务遇到循环依赖问题,解决:

AVL树在插入时保持平衡的旋转过程

目录 AVL树节点的定义 AVL树的插入 AVL树的旋转 二叉搜索树虽可以缩短查找的效率&#xff0c;但如果数据有序或接近有序二叉搜索树将退化为单支树&#xff0c;查找元素相当于在顺序表中搜索元素&#xff0c;效率低下。于是在这两位俄罗斯的数学家G.M.Adelson-Velskii 和E.M.…

《LeetCode热题100》---<6.①矩阵四道(二维数组)>

本篇博客讲解LeetCode热题100道矩阵篇中的四道题 第一道&#xff1a;矩阵置零&#xff08;中等&#xff09; 第二道&#xff1a;螺旋矩阵&#xff08;中等&#xff09; 第一道&#xff1a;矩阵置零&#xff08;中等&#xff09; 方法一&#xff1a;使用标记数组 class Solutio…

C语言指针(1)

目录 一、内存和地址 1、生活中的例子 2、内存的关系 二、指针变量和地址 1、&符号&#xff0c;%p占位符 2、一个简单的指针代码。 3、理解指针 4、解引用操作符 5、指针变量的大小。 三、指针变量类型的意义 1、指针解引用的作用 2、指针指针 3、指针-指针 4…