理解线程 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

相关文章

Java中如何正确的停掉线程

《Java中如何正确的停掉线程》Java通过interrupt()通知线程停止而非强制,确保线程自主处理中断,避免数据损坏,线程池的shutdown()等待任务完成,shutdownNow()强制中断... 目录为什么不强制停止为什么 Java 不提供强制停止线程的能力呢?如何用interrupt停止线程s

python 线程池顺序执行的方法实现

《python线程池顺序执行的方法实现》在Python中,线程池默认是并发执行任务的,但若需要实现任务的顺序执行,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋... 目录方案一:强制单线程(伪顺序执行)方案二:按提交顺序获取结果方案三:任务间依赖控制方案四:队列顺序消

SQLServer中生成雪花ID(Snowflake ID)的实现方法

《SQLServer中生成雪花ID(SnowflakeID)的实现方法》:本文主要介绍在SQLServer中生成雪花ID(SnowflakeID)的实现方法,文中通过示例代码介绍的非常详细,... 目录前言认识雪花ID雪花ID的核心特点雪花ID的结构(64位)雪花ID的优势雪花ID的局限性雪花ID的应用场景

深入理解go中interface机制

《深入理解go中interface机制》本文主要介绍了深入理解go中interface机制,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录前言interface使用类型判断总结前言go的interface是一组method的集合,不

SpringBoot实现虚拟线程的方案

《SpringBoot实现虚拟线程的方案》Java19引入虚拟线程,本文就来介绍一下SpringBoot实现虚拟线程的方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录什么是虚拟线程虚拟线程和普通线程的区别SpringBoot使用虚拟线程配置@Async性能对比H

在Java中实现线程之间的数据共享的几种方式总结

《在Java中实现线程之间的数据共享的几种方式总结》在Java中实现线程间数据共享是并发编程的核心需求,但需要谨慎处理同步问题以避免竞态条件,本文通过代码示例给大家介绍了几种主要实现方式及其最佳实践,... 目录1. 共享变量与同步机制2. 轻量级通信机制3. 线程安全容器4. 线程局部变量(ThreadL

Linux线程同步/互斥过程详解

《Linux线程同步/互斥过程详解》文章讲解多线程并发访问导致竞态条件,需通过互斥锁、原子操作和条件变量实现线程安全与同步,分析死锁条件及避免方法,并介绍RAII封装技术提升资源管理效率... 目录01. 资源共享问题1.1 多线程并发访问1.2 临界区与临界资源1.3 锁的引入02. 多线程案例2.1 为

Java Spring的依赖注入理解及@Autowired用法示例详解

《JavaSpring的依赖注入理解及@Autowired用法示例详解》文章介绍了Spring依赖注入(DI)的概念、三种实现方式(构造器、Setter、字段注入),区分了@Autowired(注入... 目录一、什么是依赖注入(DI)?1. 定义2. 举个例子二、依赖注入的几种方式1. 构造器注入(Con

Java中的xxl-job调度器线程池工作机制

《Java中的xxl-job调度器线程池工作机制》xxl-job通过快慢线程池分离短时与长时任务,动态降级超时任务至慢池,结合异步触发和资源隔离机制,提升高频调度的性能与稳定性,支撑高并发场景下的可靠... 目录⚙️ 一、调度器线程池的核心设计 二、线程池的工作流程 三、线程池配置参数与优化 四、总结:线程

WinForm跨线程访问UI及UI卡死的解决方案

《WinForm跨线程访问UI及UI卡死的解决方案》在WinForm开发过程中,跨线程访问UI控件和界面卡死是常见的技术难题,由于Windows窗体应用程序的UI控件默认只能在主线程(UI线程)上操作... 目录前言正文案例1:直接线程操作(无UI访问)案例2:BeginInvoke访问UI(错误用法)案例