Linux多线程概念

背景知识

内存管理

OS进行内存管理不是以字节为单位的,而是以内存块为单位的,默认大小为4kb;系统和磁盘文件进行IO交互的单位是4kb(8个扇区);OS对内存管理实质上是对页框进行管理。

页框(Page Frame)
页框是物理内存中的固定大小的连续块,用于存储进程的页面数据。操作系统将物理内存划分为多个大小相等的页框,作为内存管理的基本单位。例如,在x86架构中,常见页框大小为4KB。

页帧(Page Frame)
页帧与页框通常被视为同义词,均指物理内存中的固定块。但在某些上下文中,"页帧"可能更强调其作为地址映射的载体,而"页框"侧重物理划分。实际使用中两者常互换。

对此,操作系统使用struct page结构体来描述和管理页框(struct page memory[1048576])。

struct page {unsigned long flags;            // 页状态标志(如脏页、锁定位等)union {struct {                    // 页缓存和匿名页使用struct list_head lru;struct address_space *mapping;pgoff_t index;unsigned long private;};struct {                    // slab分配器使用struct kmem_cache *slab_cache;void *freelist;union {void *s_mem;unsigned long counters;};};// 其他使用场景的union分支...};atomic_t _refcount;             // 引用计数unsigned long compound_head;    // 复合页头指针unsigned char compound_dtor;    // 复合页析构函数IDunsigned char compound_order;   // 复合页阶数atomic_t compound_mapcount;     // 复合页映射计数unsigned int page_type;         // 页面类型标识void *virtual;                  // 内核虚拟地址(若非高端内存)// 其他架构相关成员...
};

页表

虚拟地址32位,系统这样划分:

因此,真实的页表是这样的,页表本质是搜索页框号(二级页表):

页目录4kb,页表一个2kb(每个2字节),一共最大4kb+2MB。根据数据类型就可以拿到完整的数据,每个进程会把自己的页目录起始地址放在CR3寄存器中,CPU使用mmu和该寄存器就可以计算出物理地址。

线程概念

基本概念

线程在进程内部执行,是CPU调度的基本单位。在一个程序里的一个执行路线就叫做线程(thread),更准确的定义是:线程是“一个进程内部的控制序列;每个进程至少一个执行线程。Linux中线程就是创建一批共享地址空间和页表的task_struct,来执行代码中的一部分任务。

进程定义的内核观点:承担分配系统资源的基本实体。

为了方便对线程进行管理,映入struct tcb结构体(Windows),但是在linux不单独设计,复用PCB表示统一执行流,这样不需要设计单独的调度算法:

struct task_struct {volatile long state;    // 线程状态(运行、就绪等)void *stack;            // 线程栈指针unsigned int flags;     // 标志位int prio;               // 动态优先级struct list_head tasks; // 线程链表节点struct mm_struct *mm;   // 内存管理信息// 更多字段...
};

Linux中的执行流都叫做轻量级进程,CPU看到的执行流<=进程。

线程独立的资源:线程ID,一组寄存器,栈,errno,信号屏蔽字,调度优先级。

线程创建

在Linux中线程叫用户级线程,Windows中叫Windows内核级线程,因为Windows真正实现这一概念。

样例代码:

Linux中没有线程只有进程,因此编译时要引入库-L pthread,系统调研只会给上层用户提供创建轻量级进程的接口,pthread库是Linux自带的原生线程库,对轻量级进程接口进行封装,按照线程接口方式交给用户。

#include<iostream>
using namespace std;
#include<unistd.h>
#include<pthread.h>
void* newthread(void * s)
{while(1){cout << "this is new thread,pid:" <<getpid()<< endl;sleep(2);}
}
int main()
{pthread_t id;pthread_create(&id, nullptr, newthread, (void *)"syx 666");while(1){sleep(1);cout << "this is main thread,pid:" <<getpid()<< endl;}return 0;
}

运行:

查看进程发现:

说明这本就是一个进程,只不过是不同执行流而已,pid一样。

查看线程:

ps -aL

这里LWP(Light Weight Thread)就是线程号。因此,OS调度时看的是LWP,LWP才是执行流的标识,main函数式主线程,其他线程是新线程。

线程优缺点

优点

cpu会把热点数据放在缓存;多线程因为会使用同一个热点数据,减少了进程间的cache切换,相比进程具有更轻量的上下文切换开销,所以调度效率高,进程切换的时候会把上个cache存储的数据删掉,重用率低。

线程上下文切换通常只需保存寄存器状态和栈指针,而进程切换还需刷新TLB、更新内存管理单元等操作。实测数据显示,线程切换开销约为进程切换的1/10到1/30,这种差异在高并发场景下尤为明显。

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

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

3.线程占用的资源要比进程少很多 能充分利用多处理器的可并行数量

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

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

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

线程不是越多越好,合适最好。计算密集型应用时线程数和核数相等最好。IO可以多创建。

缺点

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

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

3.缺乏访问控制:进程是访问控制的基本粒度,在一个线程中调用某些OS函数会对整个进程造成影响。 编程难度提高 编写与调试一个多线程程序比单线程程序困难得多

4.编写难度高。

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

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

相关文章

【Problem】动态规划之跳跃游戏系列

一、跳跃游戏 55. 跳跃游戏 - 力扣&#xff08;LeetCode&#xff09;https://leetcode.cn/problems/jump-game/description/?envTypeproblem-list-v2&envIddynamic-programming class Solution { public:bool canJump(vector<int>& nums) {// 状态定义&#x…

射频EVM

EVM&#xff08;Error Vector Magnitude&#xff0c;误差矢量幅度&#xff09;是衡量无线通信系统中调制质量的重要指标&#xff0c;尤其用于评估信号的调制误差和系统性能。它通常用来表示传输信号与理想信号之间的偏差&#xff0c;特别是在数字通信中。EVM的基本概念&#xf…

Java 更改 Word 文档中文本颜色

在日常的自动化文档处理中&#xff0c;我们经常会遇到需要对 Word 文档内容进行编程修改的需求&#xff0c;其中一项常见且重要的操作就是更改文本的颜色。无论是为了突出重点、统一品牌风格&#xff0c;还是实现动态内容展示&#xff0c;精准地修改文本颜色都是一个核心痛点。…

STM32—SPI协议

文章目录一、SPI 协议简介二、硬件电路2.1.SPI的连接2.2.数据的移位2.3.时序基本单元2.3.1.起始条件和终止条件2.3.2.模式 02.3.3.模式 12.3.4.模式 22.3.5.模式 32.4.时序三、软件实现四、W25Q644.1.简介4.2.硬件电路4.3.框图4.4.操作注意事项五、实验一、SPI 协议简介 SPI&a…

Qt中的QWebEngineView

第1章 本地目录结构1.1 自己写的两个网页(html)mermaid.html &#xff08;自己写的网页界面&#xff09;WebTest.html (自己写的网页界面)qwebchannel.js (Qt下载安装之后&#xff0c;会在安装目录下有这个文件&#xff0c;需要将安装目录下的改文件拷贝…

Flutter 应用国际化 (i18n) 与本地化 (l10n) 完整指南

Flutter 国际化 (i18n) 完全指南&#xff1a;从入门到精通 在现代移动应用开发中&#xff0c;支持多语言是触达全球用户的基本要求。Flutter 提供了强大且灵活的国际化 (i18n) 和本地化 (l10n) 支持。本文将带你从零开始&#xff0c;一步步深入掌握在 Flutter 中实现国际化的几…

计算机视觉与深度学习 | 计算机视觉中线特征提取与匹配算法综述

文章目录 一、线特征提取算法原理 1.1 Hough变换及其优化 1.2 LSD算法 1.3 EDLines算法 二、核心数学公式 2.1 直线表示与误差计算 2.2 LSD算法关键公式 三、线特征匹配算法 3.1 LBD描述符 3.2 匹配策略 四、代码实现 4.1 LSD线段检测(Python) 4.2 LBD特征匹配(C++) 五、算…

Transformer 模型:Attention is All You Need 的真正含义

2017 年&#xff0c;Google Brain 发布了一篇具有里程碑意义的论文——《Attention Is All You Need》&#xff0c;这篇论文不仅首次提出了 Transformer 模型&#xff0c;更重要的是&#xff0c;它宣称“注意机制&#xff08;Attention Mechanism&#xff09;就足以构建强大的模…

数据库约束表的设计

数据库约束概念&#xff1a;数据库约束是关系型数据库的一个重要功能&#xff0c;主要是保证数据的完整性&#xff0c;也可理解为数据的正确性&#xff08;数据本身是否正确&#xff0c;关联关系是否正确&#xff09;&#xff08;一般是用在指定列上&#xff09;常见的约束类型…

【案例分享】TeeChart 助力 Softdrill 提升油气钻井数据可视化能力

在钻井与地质工程领域&#xff0c;数据可视化是核心环节。图表不仅需要精确与高效&#xff0c;还需符合行业习惯并支持交互与定制。Softdrill 自 2012 年起在核心产品中集成了TeeChart 图表库&#xff0c;将复杂的井下数据转化为直观的工程图表&#xff0c;极大提升了钻井工程师…

【Flink】Flink Runtime 架构设计

Flink Runtime 架构设计 整体架构 ┌─────────────────────────────────────────────────────────────────┐ │ Flink Runtime │ ├─────────…

Git 命令教程

Git介绍 分布式版本控制系统。 Git命令 初始化/全局配置git init初始化一个Git仓库&#xff08;会创建一个.git的目录&#xff09;git config --global user.name “name”设置提交时的用户名git config user.name查看设置的用户名git config --global user.email “youemail.c…

git config --global user.name指令报错时的解决方案

问题分析 %HOMEDRIVE%%HOMEPATH%/.gitconfig 是Windows环境变量的表示方式&#xff1a; %HOMEDRIVE% 通常是 C:%HOMEPATH% 通常是 \Users\你的用户名完整路径应该是&#xff1a;C:\Users\你的用户名\.gitconfig 但这里环境变量没有被正确解析&#xff0c;显示的是字面意思。 …

websocket和socket io的区别

好的&#xff0c;这是一个更具体也更常见的问题。WebSocket 是一种协议&#xff0c;而 Socket.IO 是一个库&#xff0c;它使用了 WebSocket 但提供了多得多的功能。 简单比喻&#xff1a; WebSocket 就像是给你提供了一条高效的“快递专线”&#xff08;双向通信通道&#xff…

Nginx反向代理与负载均衡部署

Nginx反向代理与负载均衡部署实战指南前言一、规划部署负载均衡和反向代理二、部署Nginx负载均衡器2.1. 准备基础环境2.2. 创建Nginx运行用户2.3. 编译安装Nginx2.4. 配置Nginx系统服务2.5. 验证Nginx安装三、部署后端2台Tomcat应用服务器3.1. 安装JDK3.2. 部署Tomcat实例13.3.…

从源码和设计模式深挖AQS(AbstractQueuedSynchronizer)

AQS 概念 AbstractQueuedSynchronizer&#xff08;AQS&#xff09; 是 Java 并发包 (java.util.concurrent.locks) 的核心基础框架&#xff0c;它的实现关键是先进先出 (FIFO) 等待队列和一个用volatile修饰的锁状态status。具体实现有 : ReentrantLock、Semaphore、CountDownL…

Dart → `.exe`:Flutter 桌面与纯命令行双轨编译完全指南

Dart → .exe&#xff1a;Flutter 桌面与纯命令行双轨编译完全指南 关键词&#xff1a;Dart、Flutter、Windows、可执行文件、桌面端、CLI、交叉编译 1. 前言 很多开发者以为 Dart 只能跑在 AOT 移动端或 Web 端&#xff0c;其实 官方工具链早已支持一键输出 Windows 原生 .ex…

互联网接入网中PPPoE和PPP协议

<摘要> PPPoE和PPP是宽带接入网络中至关重要的协议组合&#xff0c;其中PPP提供通用的点对点链路层解决方案&#xff0c;而PPPoE则是在以太网架构上扩展PPP应用的技术桥梁。本文从技术演进视角系统解析了两者的内在关联与本质区别&#xff1a;PPP作为成熟链路层协议&…

详细解析SparkStreaming和Kafka集成的两种方式的区别和优劣

spark streaming是基于微批处理的流式计算引擎&#xff0c;通常是利用spark core或者spark core与spark sql一起来处理数据。在企业实时处理架构中&#xff0c;通常将spark streaming和kafka集成作为整个大数据处理架构的核心环节之一。 针对不同的spark、kafka版本&#xff0…

Kite Compositor for Mac v2.1.2 安装教程|DMG文件安装步骤(Mac用户必看)

Kite Compositor​ 是一款专为 ​macOS​ 设计的 ​轻量级界面设计 & 动画制作工具&#xff0c;它可以让你像拼图一样直观地 ​创建、编辑和预览用户界面&#xff08;UI&#xff09;以及动画效果。 一、下载文件 首先&#xff0c;你得先把这个 ​Kite Compositor for Mac …