[ Linux ] 一篇带你理解Linux下线程概念

news2025/1/18 5:39:14

目录

1.Linux线程的概念

1.1什么是线程

1.1.1如何验证一个进程内有多个线程?

1.2线程的优点

1.3线程的缺点

1.4 线程异常

1.5 线程用途

2.Linux进程与线程

2.1进程和线程

2.2 进程和线程的关系

2.3如何看待之前学习的单进程?


1.Linux线程的概念

1.1什么是线程

在之前我们谈过Linux的进程,每一个进程都有自己的PCB,和自己的进程地址空间。地址空间和物理内存通过页表建立映射。那么现在我要创建一个新的进程,操作系统要给这个新的进程创建pcb,建立自己的地址空间,页表等等。因此创建一个进程的成本是非常大的。那么现在有这样一种方法,要创建一个进程的时候我只创建PCB, 所有的pcb访问的是同一个地址空间。因此这几个进程是共享地址空间的。我们把地址空间的某一区域划分给一个tast_struct,那么这个tast_struct在地址空间中就有了自己的一小份区域。那么我们把这里的一个tast_struct的一个执行流就可以称为线程。

因此总结一下什么是线程呢?

  • 在一个程序里的一个执行路线就叫做线程(thread)。更准确的定义时:线程是“一个进程内部的控制序列”。
  • 一切进程至少都有一个执行线程
  • 线程在进程内部运行,本质是进程地址空间内运行
  • 在Linux系统中,在CPU眼中,看到的TCBP都要比传统的进程更加轻量化
  • 透过进程虚拟地址空间,可以看到进程的大部分资源,将进程资源合理分配给每个执行流,就形成了线程执行流。

在Linux当中,没有进程和线程的概念上的区分,只有一个叫做执行流!Linux的线程就是用进程PCB模拟的。CPU看到的所有的tast_struct都是一个执行流(线程)。

重要总结:

以前的进程 = 内核数据结构(PCB)+进程对应的代码和数据,而现在我们站在内核视角给进程重新下一个定义。(内核角度)进程是承担分配系统资源的基本实体(进程的基座属性)。因此Linux进程的最大意义不是被执行,而是向系统申请资源的基本单位!而之前我们所写的进程内部就只有一个执行流的进程(单执行流),而今天我们学习了进程内部可以有多个执行流,我们叫做多执行流的进程(多执行流)。

CPU视角下,tast_struct <= 传统的进程PCB. 【等于的时候就是单执行流】

因此,在Linux下,没有真正意义的线程,而是用tast_struct模拟实现的。因此把Linux下的“进程”<= 其他操作系统的进程概念。也可以称之为“轻量级进程”。

线程是调度的基本单位。进程内部有多个执行流,而一个执行流就可以称之为一个线程。一个进程内部可以有多个线程。

1.1.1如何验证一个进程内有多个线程?

Linux下没有直接创建线程的接口,但是有原生的线程库。pthread_create(不是操作系统的接口)。是程序员写出来的库。

pthread_create 线程创建

注意:编译的时候要链接这个库,因此要带 -lpthread 选项,我们可以体现在makefile之中

mythread:mythread.cc
	g++ -o $@ $^ -pthread -std=c++11
.PHONY:clean
clean:
	rm -f mythread

pthread_join线程等待

线程和进程一样也要等待,因此线程等待的接口时pthread_join。

有了这两个接口,我们就能写出最简单的线程代码,至于函数和代码内部的细节在线程控制会重点强调。

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

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

void *callback1(void *args)
{
    string name = (char*)args;
    while(true)
    {
        cout << name << endl;
        sleep(1);
    }
}
void *callback2(void *args)
{
    string name = (char*)args;
    while(true)
    {
        cout << name << endl;
        sleep(1);
    }
}

int main()
{
    pthread_t tid1;
    pthread_t tid2;

    pthread_create(&tid1, nullptr, callback1, (void *)" thread 1");
    pthread_create(&tid2, nullptr, callback2, (void *)" thread 2");

    while (true)
    {
        cout << "我是主线程............" << endl;
        sleep(1);
    }

    pthread_join(tid1, nullptr);
    pthread_join(tid2, nullptr);
    return 0;
}

我们发现果然是有3个执行流在运行。那么如何证明线程是用进程来模拟的呢?

我们使用ps axj | grep mypthread 命令来查看当前进程的状态

我们发现,我们明明有3个执行流在运行,我们只查到了一个pid为27829进程,因为我们三个执行流是三个线程,属于一个进程,那么我们如何查看线程呢,使用这条命令

ps -aL

我们看到了有3个线程在运行,和我们预期的一样,三个执行流的pid都是27829,这也验证了这三个执行流属于同一个进程,其中第一个线程的LWP(Light weight process -- 轻量级进程编号)也是27829说明是主线程。

因此我们验证了一个进程内部只有多个线程(执行流)。

1.2线程的优点

  • 创建一个新线程的代价要比创建一个新进程小得多
  • 与进程之间的切换相比,线程之间的切换需要操作系统做得工作要少很多
  • 线程占有的资源要比进程少很多
  • 能充分利用多处理器的可并行数量
  • 在等待慢速I/O操作结束的同时,程序可执行其他的计算任务
  • 计算密集型应用,为了能在多处理器系统上运行,将计算分解到多个线程中实现
  • I/O密集型应用,为了提高性能,将I/O操作重叠,线程可以同时等待不同的I/O操作

如果以上线程的有点无法理解,待线程学习完之后回头看便可理解。

1.3线程的缺点

  • 性能损失

一个很少被外部事件阻塞的计算密集型线程往往无法与共它线程共享同一个处理器。如果计算密集型线程的数量比可用的处理器多,那么可能会有较大的性能损失,这里的性能损失指的是增加了额外的同步和调度开销,而可用的资源不变。

  • 健壮性降低

编写多线程需要更全面更深入的考虑,在一个多线程程序里,因时间分配上的细微偏差或者因共享了不该共享的变量而造成不良影响的可能性是很大的,换句话说线程之间是缺乏保护的。

  • 缺乏访问控制

进程是访问控制的基本粒度,在一个线程中调用某些OS函数会对整个进程造成影响。

  • 编程难度提高

编写与调试一个多线程程序比单线程程序困难得多

1.4 线程异常

  • 单个线程如果出现除零,野指针问题导致线程崩溃,进程也会随着崩溃
  • 线程是进程的执行分支,线程出异常,就类似进程出异常,进而触发信号机制,终止进程,进程终止,该进程内的所有线程也就随即退出

1.5 线程用途

  • 合理的使用多线程,能提高CPU密集型程序的执行效率
  • 合理的使用多线程,能提高IO密集型程序的用户体验(如生活中我们一边写代码一边下载开发工具,就是多线程运行的一种表现)

2.Linux进程与线程

2.1进程和线程

  • 进程时资源分配的基本单位
  • 线程是调度的基本单位
  • 线程共享进程数据,但也拥有自己的一部分数据(线程ID,一组寄存器,栈,errno,信号屏蔽字,调度优先级)
  • 进程的多个线程共享同一地址空间,因此Text Segment(代码段),Data Segment(数据段)都是共享的,如果定义一个函数,在各线程中都可以调用,如果定义一个全局变量,在各线程中都可以访问到。除此之外,各线程还共享以下进程资源和环境(文件描述符表,每个信号的处理方式,当前工作目录,用户id和组id)

2.2 进程和线程的关系

进程和线程的关系如下图:

2.3如何看待之前学习的单进程?

之前学习的单进程是具有一个线程执行流的进程。

(本篇完)

 

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

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

相关文章

迪杰斯特拉算法求图的最短路径(java)

迪杰斯特拉算法 图的最短路径的解法 单源最短路径 从一个点开始&#xff0c;可以找到其中任意一个点的最短路径。 多源最短路径 从任何一个点开始&#xff0c;可以找到其中任何一个点的最短路径。 解题过程 给定一个带权有向图G(G, V), 另外&#xff0c;还给定 V 中的一…

力扣(LeetCode)1832. 判断句子是否为全字母句(C++)

哈希集合1 哈希集合记录 262626 个字母是否出现&#xff0c;一次遍历字符串&#xff0c;维护哈希集合&#xff0c;同时维护答案。遍历完成&#xff0c;仅当答案等于 262626 &#xff0c;句子是全字母句。 class Solution { public:bool checkIfPangram(string sentence) {boo…

轻松提高性能和并发度,springboot简单几步集成缓存

目录 1、缘由 2、技术介绍 2.1、技术调研 2.2、spring支持的cache 2.3、cache的核心注解 2.3.1 EnableCaching 2.3.2 Cacheable 2.3.3 CachePut 2.3.4 CacheEvict 2.4 cache的架构 2.5 cachemanager的实现类 3、搞个例子 3.1 为什么使用redis 作为缓存 3.2 代码走起…

【虚幻引擎】UE4/UE5数字孪生与前端Web页面匹配

一、数字孪生 数字孪生是一种多维动态的数字映射&#xff0c;可大幅提高效能。数字孪生是充分利用物理模型、传感器更新、运行历史等数据&#xff0c;集成多学科、多物理量、多尺度、多概率的仿真过程&#xff0c;在虚拟空间中完成对现实体的复制和映射&#xff0c;从而反映物理…

MySQL常用窗口函数

1、窗口函数概念 窗口的概念非常重要&#xff0c;它可以理解为记录集合&#xff0c;窗口函数也就是在满足某种条件的记录集合上执行的特殊函数对于每条记录都要在此窗口内执行函数&#xff0c;有的函数随着记录不同&#xff0c;窗口大小都是固定的&#xff0c;这种属于静态窗口…

c语言:枚举类型—enum

枚举类型一.常见形式二.枚举和宏定义三.枚举的意义四.插个小知识一.常见形式 这里举一个例子&#xff0c;我想要枚举颜色 注意一下细节&#xff0c;所有成员间用逗号隔开&#xff0c;最后一个成员后不加标点符号 这里看上去和定义结构体和联合体的样式一样&#xff0c;但其实前…

minio安装部署和minIO-Client的使用

minio安装部署和minIO-Client的使用 一、服务器安装minio 1.进行下载 下载地址&#xff1a; GNU/Linux https://dl.min.io/server/minio/release/linux-amd64/minio2.新建minio安装目录&#xff0c;执行如下命令 mkdir -p /home/minio/data把二进制文件上传到安装目录后&a…

【PAT甲级 - C++题解】1128 N Queens Puzzle

✍个人博客&#xff1a;https://blog.csdn.net/Newin2020?spm1011.2415.3001.5343 &#x1f4da;专栏地址&#xff1a;PAT题解集合 &#x1f4dd;原题地址&#xff1a; 题目详情 - 1128 N Queens Puzzle (pintia.cn) &#x1f511;中文翻译&#xff1a;皇后问题 &#x1f4e3;…

第9章 无线网络和移动网络

目录 9.1 无线局域网 WLAN 9.1.1 无线局域网的组成 1. 无线局域网 WLAN (Wireless Local Area Network) 2. IEEE 802.11 3. 移动自组网络 9.1.2 802.11 局域网的物理层 9.1.3 802.11 局域网的 MAC 层协议 1. CSMA/CA 协议 2. 时间间隔 DIFS 的重要性 3. MAC两个子层…

acwing基础课——Floyd

由数据范围反推算法复杂度以及算法内容 - AcWing 常用代码模板3——搜索与图论 - AcWing 基本思想&#xff1a; floyd算法的原理是基于动态规划的基础上实现的&#xff0c;因为是稠密图我们通过邻接矩阵来存储&#xff0c;我们将各点距离初始化为正无穷(该点到自己的距离为0)&…

软件测试基础理论体系学习8-什么是验收测试?验收测试的内容是什么?过程是什么?有什么测试策略?

8-什么是验收测试&#xff1f;验收测试的内容是什么&#xff1f;过程是什么&#xff1f;有什么测试策略&#xff1f;1 验收测试的主要内容1.1 简介和说明1.2 验收测试的目的1.3 验收测试的任务1.4 验收测试主要内容1.4.1 验收测试标准1.4.2 配置复审1.4.3 α、β测试2 验收测试…

基于intel低功耗平台边缘计算解决方案助力半导体设备升级

半导体芯片是现代电子领域的大脑。事实上&#xff0c;在通信、计算、零售、医疗保健和运输应用领域&#xff0c;半导体芯片为各种先进技术提供了基础。2020年全球半导体销售额增长6.5%&#xff0c;相关制造设备的生产需求也相应增加。 某业内日本半导体设备制造厂商&#xff0…

营销不知道怎么做,不妨试试社交新零售电商结合新型引流模式

大家好&#xff0c;我是林工&#xff0c;如今移动互联网的快速发展为社交媒体分销提供了生根发芽的土壤&#xff0c;人们不仅仅满足于单方面的购物消费&#xff0c;开始利用社交属性&#xff0c;出现了具有强大带货能力和分销能力的人群&#xff0c;也就是当时的代购和微商&…

BEVFormer-accelerate:基于EasyCV加速BEVFormer

作者&#xff1a;贺弘 夕陌 谦言 临在 导言 BEVFormer是一种纯视觉的自动驾驶感知算法&#xff0c;通过融合环视相机图像的空间和时序特征显式的生成具有强表征能力的BEV特征&#xff0c;并应用于下游3D检测、分割等任务&#xff0c;取得了SOTA的结果。我们在EasyCV开源框架&…

【MySQL自学之路】第3天——MySQL数据库服务的基本操作

目录 前言 数据库的初始化 添加环境变量 数据库配置文件的编写【my.ini】 初始化MySQL服务 MySQL服务操作 安装一个MySQL服务 删除一个MySQL服务 开启你的MySQL服务 关闭你的MySQL服务 查看电脑中存在的MySQL服务 其他操作 登录数据库 修改密码 登录跳过密码验证…

Hi3861开发遇到的小问题及解决方法和一些小技巧

开发过程中的学习 一、硬件支持开启&#xff08;替换文件&#xff09; &#x1f604; 在D:\ubuntu2004Harmony镜像中通过vscode 使用编译命令 python build.py wifiiot是存在一个小问题 No option ‘riscv32-unknown-elf-gcc_path’ in section: ndk’ ” 而在E:\鸿蒙资料\鸿蒙…

[附源码]Python计算机毕业设计儿童绘本租赁网站Django(程序+LW)

该项目含有源码、文档、程序、数据库、配套开发软件、软件安装教程 项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等…

基于jsp+mysql+ssm工程车辆租赁系统-计算机毕业设计

项目介绍 随着现代工程车租赁的快速发展&#xff0c;可以说工程车租赁已经逐渐成为现代工程车租赁过程中最为重要的部分之一。但是一直以来我国传统的工程车租赁并没有建立一套完善的行之有效的工程车租赁系统&#xff0c;传统的工程车租赁已经无法适应高速发展&#xff0c;无…

Idea制作docker镜像

一、配置docker远程访问 1.在linux系统中查找docker.service文件&#xff0c;使用命令&#xff1a;find / -name docker.service 2.编辑 /usr/lib/systemd/system/docker.service文件&#xff0c;在ExecStart/usr/bin/dockerd后面添加 -H tcp://0.0.0.0:2375 3.重新加载dock…

持续盈利背后,水滴“新增长”难寻?

配图来自Canva可画 近年来随着互联网红利消失&#xff0c;越来越多的互联网企业回归效益&#xff0c;开始将“降本增效”纳入了议事日程之中&#xff0c;这不仅体现在BAT等一众大厂的财报中&#xff0c;就连水滴等互联网领域的“后起之秀”也在争相效法。 作为过去几年有名的…