理解线程 ID 和 LWP

2024-08-24 03:28
文章标签 线程 理解 id lwp

本文主要是介绍理解线程 ID 和 LWP,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

序言

 在不同的系统中,为了更好地管理用户可能会采取不同的编号。比如在学校的教务系统中,管理学生使用的是学号;但是在住宿系统中,为了更加方便的获取一个学生的寝室信息,可能会采取结合你是哪一栋,哪一层,哪一间,哪一个铺位号的信息来作为新的的编号。
 有很多在我们计算机系统体系的设定,其实利用生活的道理都能非常直观的解释。本篇文章的内容需要一些线程的前置知识,可以点击此处查看哦 点击跳转👈 。


1. 引出问题

 在 Linux 系统中,我们可以通过指令 ps -a 来查看当前用户下的启动程序的进程的 PID,同时我们在代码中使用函数 getpid() 函数也可以获取当前进程的 PID,举个栗子:

1 Test.cc                                                                                           X 1 #include <iostream>2 #include <unistd.h>3 4 // 一段非常简单的程序5 // 每隔两秒打印当前进程的PID6 7 int main()8 {9     while(true)10     {11         std::cout << "Hello, my pid is " << getpid() << "." << std::endl;12         sleep(2);13     }14 15     return 0;16 }

启动这个程序之后,我们再使用该指令来获取当前进程的信息:
在这里插入图片描述
可以很明显的看到,两者的 PID 是一样的。

 现在,我们来回忆一下线程的部分知识点:

  • 线程是进程中的一个实体,是 CPU 调度和分派的基本单位,它是 进程中的实际运作单位
  • Linux 系统上,并不存在真正意义上的线程‌,他的线程是通过 模拟进程来实现的,被称为 轻量级进程(LWP)

我们可以通过指令 ps -aL 查看当前用户下的线程消息,运行上段程序:

在这里插入图片描述
可以看到,当前程序我们并没有显示的创建副线程,但是还是存在相应线程信息,说明一个进程默认是有一个线程的(称为 主线程)。

 我们对上段程序进行一点改造,创建一个副线程执行其他任务:

#include <iostream>
#include <unistd.h>
#include <pthread.h>void *thread_Func(void *arg)
{while(true){std::cout << "Hello, I am pthread_1, my tid is " << pthread_self() << "." << std::endl;sleep(2);}return nullptr;
}int main()
{pthread_t tid = 0;pthread_create(&tid, nullptr, thread_Func, nullptr);while(true){std::cout << "Hello, I am main pthread, my tid is " << pthread_self() << "." << std::endl;sleep(2);}pthread_join(tid, nullptr);return 0;
}

创建了一个父线程和主线程分别执行任务,现在再使用指令 ps -aL
在这里插入图片描述
通过打印出的结果,我们可以得出一下信息:

  • 主线程的 LWP 和该进程的 PID 是相同的
  • 一个进程包含的线程的 LWP 是连续的
  • 线程的 LWPTID 并不相同

 前面铺垫了这么多内容,终于是引出了疑问,我们进程使用指令和函数得出的 PID 是一个值,为什么线程的 LWP 和通过函数 pthread_self() 函数得出的值不一样呢?


2. 底层设计

2.1 LWP

LWP 轻量级进程 是在 Linux 系统下设计线程的方式,线程和进程一样也是需要管理的,所以为了更好地管理所有线程,系统存在一个结构体存储线程地相关信息,而 LWP 对应的值就是该线程对应结构体唯一的标识符。

2.2 TID

 那 TID 又代表什么呢?TID 代表一个地址!
 首先我们需要了解,Linux 的线程又叫做 轻量级进程,所以所维护的线程信息都是针对进程的角度,所以光靠这些信息是不够维护一个线程的。
 但是大家不要忘记我们使用了一个库 POSIX 线程库,该库除了为我们提供相关的线程使用方法,还帮我们维护线程所特有的属性(线程栈,线程局部存储等等)。
 现在深入内容之前,我们有必要简单回顾一下加载动态库到进程地址空间:

  1. 当程序使用到一个动态库时,会先查看他是否已经被加载到内存中了,如果没有,则先会加载到内存中
  2. 其次,会将该库映射到进程地址空间的共享区上,通过页表建立虚拟地址和物理地址的映射关系
  3. 最后,我们就可以调用共享库的函数了

当我们使用库函数 pthread_create() 创建一个线程时:
在这里插入图片描述
会在该库所在的区域为我们申请一个内存块,该块中存放对应线程的信息数据,该函数不是有一个返回型参数 tid 吗,该参数就是 线程对应内存块的地址!所以,我们可以使用该 TID 来控制对应线程!


3. 解决问题

 所以,通过基本了解了底层的情况,现在我们在来回答这个问题,LWP 是线程在内核中的结构,他被视为一个轻量级进程;TID 是线程在库中的结构,他包含线程所特有的数据和属性。


4. 总结

 在这篇文章中介绍了线程 IDLWP,以及造成不同的原因,以及底层细节,希望大家有所收获。

这篇关于理解线程 ID 和 LWP的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1101313

相关文章

Springboot3+将ID转为JSON字符串的详细配置方案

《Springboot3+将ID转为JSON字符串的详细配置方案》:本文主要介绍纯后端实现Long/BigIntegerID转为JSON字符串的详细配置方案,s基于SpringBoot3+和Spr... 目录1. 添加依赖2. 全局 Jackson 配置3. 精准控制(可选)4. OpenAPI (Spri

Java中实现线程的创建和启动的方法

《Java中实现线程的创建和启动的方法》在Java中,实现线程的创建和启动是两个不同但紧密相关的概念,理解为什么要启动线程(调用start()方法)而非直接调用run()方法,是掌握多线程编程的关键,... 目录1. 线程的生命周期2. start() vs run() 的本质区别3. 为什么必须通过 st

Linux实现线程同步的多种方式汇总

《Linux实现线程同步的多种方式汇总》本文详细介绍了Linux下线程同步的多种方法,包括互斥锁、自旋锁、信号量以及它们的使用示例,通过这些同步机制,可以解决线程安全问题,防止资源竞争导致的错误,示例... 目录什么是线程同步?一、互斥锁(单人洗手间规则)适用场景:特点:二、条件变量(咖啡厅取餐系统)工作流

Java中常见队列举例详解(非线程安全)

《Java中常见队列举例详解(非线程安全)》队列用于模拟队列这种数据结构,队列通常是指先进先出的容器,:本文主要介绍Java中常见队列(非线程安全)的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录一.队列定义 二.常见接口 三.常见实现类3.1 ArrayDeque3.1.1 实现原理3.1.2

SpringBoot3中使用虚拟线程的完整步骤

《SpringBoot3中使用虚拟线程的完整步骤》在SpringBoot3中使用Java21+的虚拟线程(VirtualThreads)可以显著提升I/O密集型应用的并发能力,这篇文章为大家介绍了详细... 目录1. 环境准备2. 配置虚拟线程方式一:全局启用虚拟线程(Tomcat/Jetty)方式二:异步

如何解决Druid线程池Cause:java.sql.SQLRecoverableException:IO错误:Socket read timed out的问题

《如何解决Druid线程池Cause:java.sql.SQLRecoverableException:IO错误:Socketreadtimedout的问题》:本文主要介绍解决Druid线程... 目录异常信息触发场景找到版本发布更新的说明从版本更新信息可以看到该默认逻辑已经去除总结异常信息触发场景复

MySQL查看表的最后一个ID的常见方法

《MySQL查看表的最后一个ID的常见方法》在使用MySQL数据库时,我们经常会遇到需要查看表中最后一个id值的场景,无论是为了调试、数据分析还是其他用途,了解如何快速获取最后一个id都是非常实用的技... 目录背景介绍方法一:使用MAX()函数示例代码解释适用场景方法二:按id降序排序并取第一条示例代码解

使用雪花算法产生id导致前端精度缺失问题解决方案

《使用雪花算法产生id导致前端精度缺失问题解决方案》雪花算法由Twitter提出,设计目的是生成唯一的、递增的ID,下面:本文主要介绍使用雪花算法产生id导致前端精度缺失问题的解决方案,文中通过代... 目录一、问题根源二、解决方案1. 全局配置Jackson序列化规则2. 实体类必须使用Long封装类3.

spring IOC的理解之原理和实现过程

《springIOC的理解之原理和实现过程》:本文主要介绍springIOC的理解之原理和实现过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、IoC 核心概念二、核心原理1. 容器架构2. 核心组件3. 工作流程三、关键实现机制1. Bean生命周期2.

SpringBoot整合mybatisPlus实现批量插入并获取ID详解

《SpringBoot整合mybatisPlus实现批量插入并获取ID详解》这篇文章主要为大家详细介绍了SpringBoot如何整合mybatisPlus实现批量插入并获取ID,文中的示例代码讲解详细... 目录【1】saveBATch(一万条数据总耗时:2478ms)【2】集合方式foreach(一万条数