深入Linux轻量级进程管理:线程创建、线程ID解析与进程地址空间页表探究

本文主要是介绍深入Linux轻量级进程管理:线程创建、线程ID解析与进程地址空间页表探究,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

🍑个人主页:Jupiter.
🚀 所属专栏:Linux从入门到进阶
欢迎大家点赞收藏评论😊

在这里插入图片描述

在这里插入图片描述

目录

  • `🚲Linux线程控制`
    • `🐏POSIX线程库`
    • `🐕创建线程`
      • `🐟指令查看轻量级进程`
          • `指令:ps -aL`
    • `🐒线程ID及进程地址空间布局`
            • *pthread_t 到底是什么类型呢?*
      • `🦔__thread与线程的局部存储`


🚲Linux线程控制

🐏POSIX线程库

在Linux中,没有线程的概念,只有轻量级进程的概念,所以Linux系统只提供了轻量级进程的系统调用,没有线程相关的系统调用,但是用户不知道轻量级进程的概念,只知道线程和进程,所以Linux将轻量级进程的系统调用进行封装,转成线程相关的接口语义提供给用户,就形成了pthread库

  • 与线程有关的函数构成了一个完整的系列,绝大多数函数的名字都是以“pthread_”开头的。
  • 要使用这些函数库,要通过引入头文<pthread.h>
  • 链接这些线程函数库时要使用编译器命令“-lpthread”选项。

🐕创建线程

  • 功能:创建一个新的线程

  • 原型
    int pthread_create(pthread_t *thread, const pthread_attr_t *attr, void *(*start_routine)(void*), void *arg);

  • 参数

    • thread:返回线程ID
    • attr:设置线程的属性,attr为NULL表示使用默认属性
    • start_routine:是个函数地址,线程启动后要执行的函数
    • arg:传给线程启动函数的参数(可以是对象等等)
    • 返回值:成功返回0;失败返回错误码
  • 错误检查:

    • 传统的一些函数是,成功返回0,失败返回-1,并且对全局变量errno赋值以指示错误。
    • pthreads函数出错时不会设置全局变量errno(而大部分其他POSIX函数会这样做)。而是将错误代码通过返回值返回pthreads同样也提供了线程内的errno变量,以支持其它使用errno的代码。对于pthreads函数的错误,建议通过返回值判定,因为读取返回值要比读取线程内的errno变量的开销更小。

注意:新线程和主线程谁先运行?这是未知的。

示例代码:

#include <iostream>
#include <unistd.h>
#include <pthread.h>void *routine(void *name)
{std::cout << "i am new thread,name:" << (char *)name << std::endl;sleep(1);return nullptr;
}
int main()
{pthread_t tid;pthread_create(&tid, nullptr, routine, (void *)("thread-1"));while (true){std::cout << "i am process,pid:" << getpid() << std::endl;sleep(1);}return 0;
}

运行结果:

🐟指令查看轻量级进程

指令:ps -aL

根据上面的结果可知:他们的pid一样,所以属于同一个进程,而且可知:OS在进行调度的时候,用的是LWP进行调度的。(因为pid可能一样,不能唯一性标识)

🐒线程ID及进程地址空间布局

  • pthread_ create函数会产生一个线程ID,存放在第一个参数指向的地址中。

  • 前面讲的线程ID属于进程调度的范畴。因为线程是轻量级进程是操作系统调度器的最小单位,所以需要一个数值来唯一表示该线程。

  • pthread_ create函数第一个参数指向一个虚拟内存单元,该内存单元的地址即为新创建线程的线程ID,属于NPTL线程库的范畴。线程库的后续操作,就是根据该线程ID来操作线程的。

线程库NPTL提供了pthread_ self函数,可以获得线程自身的ID

  • pthread_t pthread_self(void);
pthread_t 到底是什么类型呢?
  • 取决于实现。对于Linux目前实现的NPTL实现而言,pthread_t类型的线程ID本质就是一个进程地址空间上的一个地址

  • 线程的管理库维护,即描述线程的结构体与管理线程的数据结构(类似于数组)就在库里面的,其中pthread_t tid即是描述tid线程结构体的起始地址

  • 线程的栈与局部存储是在库中维护的。虽然栈是线程独立的,但是其他线程也是可以访问的。

示例代码:

#include <iostream>
#include <unistd.h>
#include <pthread.h>
#include <string.h>
#include <stdio.h>
std::string ToHex(pthread_t tid)   //将id转换为十六进制
{char buff[64];snprintf(buff, 64, "0x%lx", tid);return buff;
}void *routine(void *name)
{while (true){sleep(1);std::cout << "i am new thread,name:" << (char *)name << "  my tid is :" << ToHex(pthread_self()) << std::endl;}return nullptr;
}
int main()
{pthread_t tid;pthread_create(&tid, nullptr, routine, (void *)("thread-1"));while (true){std::cout << "i am process,pid:" << getpid() << "  new thread id:" << ToHex(tid) << std::endl;sleep(1);}return 0;
}

运行结果:
在这里插入图片描述

🦔__thread与线程的局部存储

  • __thread 是 C 和 C++ 中用于声明线程局部存储的一个关键字(或属性,具体取决于编译器和平台)。它告诉编译器或链接器,被 __thread 修饰的变量是每个线程私有的,即每个线程都有该变量的一个独立实例,这些实例之间互不影响。如:__thread int tls_variable = 0; 一般对于全局变量使用。
    示例代码:
#include <pthread.h>  
#include <stdio.h>  // 声明一个线程局部变量  
__thread int tls_variable = 0;  void* thread_function(void* arg) 
{  tls_variable = (int)arg; // 每个线程都会修改它自己的 tls_variable 副本  printf("Thread %ld has tls_variable = %d\n", (long)pthread_self(), tls_variable);  return NULL;  
}  int main() {  pthread_t threads[2];  // 创建两个线程,每个线程将接收到一个不同的参数  pthread_create(&threads[0], NULL, thread_function, (void*)1);  pthread_create(&threads[1], NULL, thread_function, (void*)2);  // 等待两个线程完成  pthread_join(threads[0], NULL);  pthread_join(threads[1], NULL);  return 0;  
}

这篇关于深入Linux轻量级进程管理:线程创建、线程ID解析与进程地址空间页表探究的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


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

相关文章

Spring组件实例化扩展点之InstantiationAwareBeanPostProcessor使用场景解析

《Spring组件实例化扩展点之InstantiationAwareBeanPostProcessor使用场景解析》InstantiationAwareBeanPostProcessor是Spring... 目录一、什么是InstantiationAwareBeanPostProcessor?二、核心方法解

深入解析 Java Future 类及代码示例

《深入解析JavaFuture类及代码示例》JavaFuture是java.util.concurrent包中用于表示异步计算结果的核心接口,下面给大家介绍JavaFuture类及实例代码,感兴... 目录一、Future 类概述二、核心工作机制代码示例执行流程2. 状态机模型3. 核心方法解析行为总结:三

Java进程CPU使用率过高排查步骤详细讲解

《Java进程CPU使用率过高排查步骤详细讲解》:本文主要介绍Java进程CPU使用率过高排查的相关资料,针对Java进程CPU使用率高的问题,我们可以遵循以下步骤进行排查和优化,文中通过代码介绍... 目录前言一、初步定位问题1.1 确认进程状态1.2 确定Java进程ID1.3 快速生成线程堆栈二、分析

springboot项目中使用JOSN解析库的方法

《springboot项目中使用JOSN解析库的方法》JSON,全程是JavaScriptObjectNotation,是一种轻量级的数据交换格式,本文给大家介绍springboot项目中使用JOSN... 目录一、jsON解析简介二、Spring Boot项目中使用JSON解析1、pom.XML文件引入依

Python中文件读取操作漏洞深度解析与防护指南

《Python中文件读取操作漏洞深度解析与防护指南》在Web应用开发中,文件操作是最基础也最危险的功能之一,这篇文章将全面剖析Python环境中常见的文件读取漏洞类型,成因及防护方案,感兴趣的小伙伴可... 目录引言一、静态资源处理中的路径穿越漏洞1.1 典型漏洞场景1.2 os.path.join()的陷

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

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

Python中bisect_left 函数实现高效插入与有序列表管理

《Python中bisect_left函数实现高效插入与有序列表管理》Python的bisect_left函数通过二分查找高效定位有序列表插入位置,与bisect_right的区别在于处理重复元素时... 目录一、bisect_left 基本介绍1.1 函数定义1.2 核心功能二、bisect_left 与

Macos创建python虚拟环境的详细步骤教学

《Macos创建python虚拟环境的详细步骤教学》在macOS上创建Python虚拟环境主要通过Python内置的venv模块实现,也可使用第三方工具如virtualenv,下面小编来和大家简单聊聊... 目录一、使用 python 内置 venv 模块(推荐)二、使用 virtualenv(兼容旧版 P

C#代码实现解析WTGPS和BD数据

《C#代码实现解析WTGPS和BD数据》在现代的导航与定位应用中,准确解析GPS和北斗(BD)等卫星定位数据至关重要,本文将使用C#语言实现解析WTGPS和BD数据,需要的可以了解下... 目录一、代码结构概览1. 核心解析方法2. 位置信息解析3. 经纬度转换方法4. 日期和时间戳解析5. 辅助方法二、L

Spring中管理bean对象的方式(专业级说明)

《Spring中管理bean对象的方式(专业级说明)》在Spring框架中,Bean的管理是核心功能,主要通过IoC(控制反转)容器实现,下面给大家介绍Spring中管理bean对象的方式,感兴趣的朋... 目录1.Bean的声明与注册1.1 基于XML配置1.2 基于注解(主流方式)1.3 基于Java