Linux的学习之路:21、线程(1)

ops/2024/12/23 1:30:06/

摘要:

本章说一下线程

目录

摘要:

一、回忆一下

二、如何理解线程

三、命令行看线程

四、利用函数进行使用

五、本章总结

1、线程的优点

2、线程的缺点

3、线程的异常

4、线程的用途


一、回忆一下

1、exe就是一个文件

2、我们的可执行程序本来就是按照空间方式进行编译的

3、可执行程序,其实就是按照区域也就是已经被划分成了以4kb为单位的大小

六字真言:先描述在组织,如下图就是先描述在组织,产生的stack_struct的结构体,这个存着虚拟地址,然后虚拟地址存着用户级页表,然后页表里面存着物理内存,然后物理内存存在磁盘内。

 那么数据如果很多该怎么存入呢?如下图在32位环境下,是通过前面十位0就是指向一级页表,然后中间十位指向二级页表,其中的一级页表也指向二级页表,然后两个都一样就是二级页表,然后二级页表指向物理内存,这时找到了变量存在的地址,然后后面12位就是指向物理内存的偏移地址,所以就可以得出

page start addr + 页内偏移

通过一级页表+二级页表找到对应的页,再通过最后12位找到准确的地址,也就是偏移量

二、如何理解线程

通过一定的技术手段,将当前进程的资源,以一定的方式划给不用的task_struct,不去申请地址空间,需要什么资源就直接向主线程要,线程在内部执行,是OS调度的基本单位

这个是Linux特有的方案:没有独立的结构体

在内核视角中:

进程:承担分配系统资源的基本实体

用户视角:该进程对应的代码+资源

内部只有一个执行流的进程,但是内部具有各个执行的进程就是线程

在CPU视角中:CPU其实不怎么关心是进程还是线程这样的概念,只认识tack_struct,CPU调度的基本单位就是“线程”

在Linux下的进程:统一称之为轻量级进程,Linux并不能直接给我们提供线程的接口,只能提位轻量级进程的接口,在用户层实现了一套用户层多线程方案,以库的方式提供给用户进行使用,pathred线程库——原生线程库

三、命令行看线程

如下图利用man手册进行查看指令,pthread_create这个函数就是创建一个线程,他的参数就是下面四个

1、pthread_t *thread:这是一个指向 pthread_t 类型变量的指针,用于接收新创建的线程的标识符。

2、const pthread_attr_t *attr:这是一个指向 pthread_attr_t 类型变量的指针,用于指定线程的属性。如果设置为 NULL,则使用默认属性。

3、void *(*start_routine) (void *):这是一个指向函数的指针,该函数将作为新线程的起始执行点。这个函数通常被称为线程函数或启动例程。它接受一个 void 指针作为参数,并返回一个 void 指针。

4、void *arg:这是一个 void 指针,用于向线程函数传递参数。

如下方代码 可以看出pid都是一个,代码如下

#include <iostream>
#include <sys/types.h>
#include <unistd.h>
#include <pthread.h>
using namespace std;void *start_routine(void *arg)
{while (1){cout << "这时新进程pid: " << getpid() << " " << endl;sleep(1);}
}int main()
{pthread_t tid;pthread_create(&tid, nullptr, start_routine, (void *)tid);while (1){cout << "这时主进程,pid:" << getpid() << endl;sleep(1);}return 0;
}

四、利用函数进行使用

如下方代码所示就是当线程结束了,然后著进行进行打印新线程

__thread int g_val = 0;void *threadRoutine(void *args)
{pthread_detach(pthread_self());while(true){cout << (char*)args << " : " << g_val << " &: " << &g_val << endl;sleep(1);}pthread_exit((void*)11);
}int main()
{pthread_t tid; pthread_create(&tid, nullptr, threadRoutine, (void *)"thread 1");while(true){cout << "main thread" << " : " << g_val << " &: " << &g_val << endl;sleep(1);break;}int n = pthread_join(tid, nullptr);cout << "n :" << n << "errstring: " << strerror(n) << endl;return 0;
}

五、本章总结

1、线程的优点

1、创建一个新线程的代价要比创建一个新进程小得多

2、与进程之间的切换相比,线程之间的切换需要操作系统做的工作要少很多

3、线程占用的资源要比进程少很多

4、能充分利用多处理器的可并行数量

5、在等待慢速I/O操作结束的同时,程序可执行其他的计算任务

6、计算密集型应用,为了能在多处理器系统上运行,将计算分解到多个线程中实现

7、I/O密集型应用,为了提高性能,将I/O操作重叠。线程可以同时等待不同的I/O操作

2、线程的缺点

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

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

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

4、编程难度提高
编写与调试一个多线程程序比单线程程序困难得多

3、线程的异常

1、单个线程如果出现除零,野指针问题导致线程崩溃,进程也会随着崩溃

2、线程是进程的执行分支,线程出异常,就类似进程出异常,进而触发信号机制,终止进程,进程终止,该进程内的所有线程也就随即退出

4、线程的用途

1、合理的使用多线程,能提高CPU密集型程序的执行效率

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

5、进程和线程

1、进程是资源分配的基本单位

2、线程是调度的基本单位

3、线程共享进程数据,但也拥有自己的一部分数据:

        线程ID
        一组寄存器
        栈
        errno       
        信号屏蔽字
        调度优先级

4、进程的多个线程共享 同一地址空间,因此Text Segment、Data Segment都是共享的,如果定义一个函数,在各线程中都可以调用,如果定义一个全局变量,在各线程中都可以访问到,除此之外,各线程还共享以下进程资源和环境:

文件描述符表
每种信号的处理方式(SIG_ IGN、SIG_ DFL或者自定义的信号处理函数)
当前工作目录
用户id和组id


http://www.ppmy.cn/ops/22020.html

相关文章

Kotlin语法入门-密封类和密封接口(11)

Kotlin语法入门-密封类和密封接口(11) 文章目录 Kotlin语法入门-密封类和密封接口(11)十一、密封类和密封接口1、密封类2、密封接口 十一、密封类和密封接口 1、密封类 在Kotlin中&#xff0c;密封类&#xff08;Sealed Class&#xff09;是一种特殊的类&#xff0c;用于表示受…

如何使用 Internet Download Manager (IDM) 来加速和优化你的下载体验 IDM 6.41下载神器

在当今信息爆炸的时代&#xff0c;下载文件和媒体内容已成为我们日常生活的一部分。无论是工作学习还是娱乐休闲&#xff0c;我们都需要从互联网上下载各种资源。为了提高下载效率和确保文件完整性&#xff0c;选择一款优秀的下载管理软件至关重要。Internet Download Manager …

R: 阿尔法α多样性计算和箱图制作,以及差异分析

# install.packages("vegan") library(vegan) library(ggplot2) library(ggpubr)setwd("xxx") # 使用read.table()函数读取数据 df <- read.table("xxx", header TRUE, row.names 1)# 转置数据框 df <- t(df)# 计算每个样品的香农多样性…

excel中怎么用乘法、加法来替代AND和OR函数

你可以使用乘法和加法来替代Excel中的AND和OR函数&#xff0c;虽然这样做可能会增加公式的复杂度&#xff0c;但在某些情况下是可行的。 1. 使用乘法替代AND函数&#xff1a;AND函数用于判断一系列条件是否同时成立&#xff0c;如果所有条件都为TRUE&#xff0c;则返回TRUE&…

安装JAVA和java IDEA并汉化过程

1.安装java: 打开java的下载链接&#xff1a; Java Downloads | Oracle 然后选择对应的版本下载即可&#xff0c;我这里是windows 所以下载这个 然后正常一步步安装即可。 2.配置java环境&#xff1a; 在桌面右键此电脑然后点击属性——高级系统设置——环境变量——然后…

模拟实现顺序表的增删改查

1.什么是顺序表&#xff1f; 顺序表是用一段物理地址连续的存储单元依次存储数据元素的线性结构&#xff0c;一般情况下采用数组存储。在数组上完成 数据的增删查改。 2.实现思路 3.实现代码 &#xff08;1&#xff09;接口的实现 public interface IList {// 新增元素,默认在…

MacPro(M2芯片)Java开发和常用工具开源软件合集

Java开发软件 1.1 IDE idea idea: https://www.jetbrains.com.cn/idea/download/?sectionmac Vs Code 链接: https://code.visualstudio.com/#alt-downloads 常用工具

【Python数据库】MongoDB

文章目录 [toc]数据插入数据查询数据更新数据删除 个人主页&#xff1a;丷从心 系列专栏&#xff1a;Python数据库 学习指南&#xff1a;Python学习指南 数据插入 from pymongo import MongoClientdef insert_data():mongo_client MongoClient(hostlocalhost, port27017)co…