【Linux】深入理解线程控制
个人主页~
深入理解线程控制
- 一、线程等待的原理
- 二、线程的局部存储
- 三、初步理解线程互斥
- 1、互斥的概念
- 2、需要互斥的原因
一、线程等待的原理
pthread_join
的作用是线程等待,其中retval
参数传递线程退出状态的原理是:当目标线程结束时,pthread_join
会将目标线程的退出状态(即线程函数的返回值或 pthread_exit
传递的参数)存储在 *retval
所指向的内存位置,也就是说,pthread_join
会修改 retval
所指向的那个 void *
类型变量的值
#include <iostream>
#include <unistd.h>
#include <pthread.h>using namespace std;int g_val = 100;void *threadRoutine(void *args)
{//参数是线程名字,转化成字符串const char *name = (const char *)args;int cnt = 5;while (true){//线程打印线程pid,以及全局变量g_val和它的地址printf("%s, pid: %d, g_val: %d, &g_val: 0X%p\n", name, getpid(),g_val, &g_val);sleep(1);cnt--;if (cnt == 0)break;}//线程退出,返回指针100pthread_exit((void *)100);
}int main()
{pthread_t pid;//主线程id,线程属性设为无,新线程函数,新线程参数pthread_create(&pid, nullptr, threadRoutine, (void *)"Thread 1");void *ret;//等待新线程结束,获得新线程的返回值pthread_join(pid, &ret);//打印线程返回值,这里强转为long long int是因为我的Linux是64位//指针是八字节大小,long long int是八字节cout << "main thread quit..., Thread 1 return val: " << (long long int)ret << endl;return 0;
}
这给我们证明了,新线程的输出型参数是可以被主线程取到的,并且全局变量是可以被所有线程访问的,是共享资源,所以全局函数也是可以被所有线程访问的
&ret接受退出状态的具体过程:
当调用 pthread_join
时,pthread_join
会阻塞当前线程,直到由 thread
参数指定的目标线程终止,一旦目标线程终止,pthread_join
会将该线程调用 pthread_exit
时传递的 void*
指针(即退出状态)赋值给 &ret
所指向的 void*
变量,即ret
,pthread_join
成功完成等待和状态获取后,会返回 0,表示操作成功,当前线程可以继续执行后续代码
二、线程的局部存储
全局变量是被所有线程共享的,如果我们的线程需要有自己的私有的东西,也就是只能够自己访问,其他线程不能访问的,我们可以在全局变量前加关键字__thread
修饰,这是编译器为我们提供的只能用来修饰内置类型的关键字
#include <iostream>
#include <pthread.h>
#include <vector>
#include <string>
#include <unistd.h>using namespace std;#define NUM 3int *p = nullptr;
//线程局部存储
__thread int val = 100;class ThreadInfo
{
public:ThreadInfo(const string &threadname):threadname_(threadname){}public:string threadname_;
};string toHex(pthread_t tid)
{char buffer[64];snprintf(buffer, sizeof(buffer), "%p", tid);return buffer;
}void *threadroutine(void *args)
{int i = 0;ThreadInfo *ti = static_cast<ThreadInfo*>(args);//线程循环,每次打印线程名称、线程ID、进程ID、被修饰变量val以及val地址while(i < 10){cout << ti->threadname_.c_str() << " is running, tid: " << toHex(pthread_self()) << ", pid: " << getpid() << ", val: " << val << ", &val: " << &val << endl;i++;val++;usleep(10000);}delete ti;return nullptr;
}int main()
{vector<pthread_t> tids;for(int i = 0; i < NUM; i++){pthread_t tid;ThreadInfo *ti = new ThreadInfo("Thread-"+to_string(i));pthread_create(&tid, nullptr, threadroutine, ti);tids.push_back(tid);usleep(1000);}//线程等待for(auto tid:tids){pthread_join(tid, nullptr);}return 0;
}
我们通过观察可以发现,在相同线程的情况下,val
的值是递增的,但对于不同的线程之间val
值是没有关系的,所以我们就通过关键字__thread
实现了线程的局部存储,这些属于每个线程的val
的地址在线程的独立栈中
三、初步理解线程互斥
1、互斥的概念
- 临界资源:多线程执行流共享的资源叫做临界资源
- 临界区:每个线程内部,访问临界资源的代码
- 互斥:任何时刻,有且只有一个执行流进入临界区,访问临界资源(对临界资源起保护作用)
- 原子性:不会被任何调度机制打断的操作,是不可再分隔的动作,该操作只有两种状态,一是完成,二是未完成(早期化学中,原子是组成物质的最小的不可分割的单位,在这样的背景下提出的原子性)
在大部分情况下,线程使用的数据都是局部变量,变量的地址空间在线程栈空间内,这种情况,变量属于单个线程,其他线程无法获得这个变量,但有时候,很多变量都需要在线程下共享,这样的变量被叫做共享变量,可以通过数据的共享,完成线程之间的交互
2、需要互斥的原因
在各个线程访问共享变量的时候,会出现多进程并发的操作,可能会带来一些问题
下面是一个经典的抢票问题,每个线程访问到共享资源的票数就给它减一,就相当于是抢走一张票
#include <iostream>
#include <cstdio>
#include <cstring>
#include <vector>
#include <unistd.h>
#include <pthread.h>using namespace std;
//四个线程一起抢票
#define NUM 4class threadData
{
public:threadData(int number){threadname = "thread-" + to_string(number);}public:string threadname;
};
//一次放出的票数
int tickets = 1000; void *getTicket(void *args)
{threadData *td = static_cast<threadData *>(args);const char *name = td->threadname.c_str();while (true){if(tickets > 0){usleep(1000);//提示出是谁抢了票,以及抢到的票号printf("who=%s, get a ticket: %d\n", name, tickets);tickets--;}elsebreak;}printf("%s ... quit\n", name);return nullptr;
}int main()
{vector<pthread_t> tids;vector<threadData *> thread_datas;for (int i = 1; i <= NUM; i++){pthread_t tid;threadData *td = new threadData(i);thread_datas.push_back(td);//这里最后一个参数因为下标从0开始,而我们的i是从1开始的,所以i-1pthread_create(&tid, nullptr, getTicket, thread_datas[i - 1]);tids.push_back(tid);}for (auto thread : tids){pthread_join(thread, nullptr);}for (auto td : thread_datas){delete td;}return 0;
}
我们将形成的程序执行两遍
第一遍:
第二遍:
我们发现,抢票怎么还能抢出第0票呢,甚至还有-1、-2票?而且竟然还有抢到一张票的情况,下面我们来详解一下
首先,如果我们只讨论一个线程,整个抢票的过程就是,ticket
在内存中,线程读取ticket
,然后线程把ticket
变量放到CPU上,CPU进行--
操作,然后再放回内存中,将原来的值覆盖
我们这么说,这个过程是不是变得很慢了呢,所以在我们读取ticket
之后,其他线程也来读取了,最后我们执行一圈后,如果他们都是一起执行完的,那么原来1000的值就变成了999,他们都抢到了第1000张票,这就是重复抢到同一张票的原因
出现负数也是这个原因,只不过不是同一时间做出返回内存的行为,在CPU进行计算的时候,要重新读取数据,如果开始时所有线程都ticket==1
,判断这里就能过得去,然后一个线程拿到了最后一张票1,其他三个线程就拿到了“假票”0
、-1
、-2
,这就是我们要进行进程互斥的原因
今日分享就到这里啦~
相关文章:
【Linux】深入理解线程控制
个人主页~ 深入理解线程控制 一、线程等待的原理二、线程的局部存储三、初步理解线程互斥1、互斥的概念2、需要互斥的原因 一、线程等待的原理 pthread_join的作用是线程等待,其中retval参数传递线程退出状态的原理是:当目标线程结束时,pthr…...
android面试情景题详解:android如何处理断网、网络切换或低速网络情况下的业务连续性
在移动互联网时代,Android应用已经成为人们日常生活中不可或缺的一部分。从社交媒体到在线购物,从移动办公到娱乐消费,几乎所有的服务都依赖于网络连接。然而,网络环境并非总是稳定可靠。断网、网络切换(如从Wi-Fi切换…...
基于 Qt 的 BMP 图像数据存取至 SQLite 数据库的实现
基于 Qt 的 BMP 图像数据存取至 SQLite 数据库的实现说明 本项目通过 Qt 框架实现了将 BMP 图像文件以二进制形式存入 SQLite 数据库,并可从数据库中读取还原为 BMP 图像文件的功能,适用于需要图像与结构化数据统一管理的场景。 整个流程分为两个主要部…...
Melos 发布pub.dev
🧰 Melos 全套实用教程(Flutter Monorepo 管理神器) 🚀 目录: 什么是 Melos?适合哪些项目? 安装与初始化 项目结构推荐 melos.yaml 配置详解 常用命令讲解(bootstrap、run、exe…...
基于 SSM 高校二手交易平台
收藏关注不迷路!! 🌟文末获取源码数据库🌟 感兴趣的可以先收藏起来,还有大家在毕设选题(免费咨询指导选题),项目以及论文编写等相关问题都可以给我留言咨询,希望帮助更多…...
Missashe考研日记-day20
Missashe考研日记-day20 1 高数 学习时间:2h30min学习内容: 今天当然是刷题啦,做不等式的证明板块的真题,证明题懂的都懂,难起来是真的一点思路都没有,这个板块还没做完,做完再总结题型。 2…...
Spring 单元测试核心注解全解:@InjectMocks、@MockBean、@Mock、@Autowired 的区别与实战
在编写 Spring Boot 应用的单元测试过程中,@InjectMocks、@MockBean、@Mock 和 @Autowired 是最常用的几个注解,但它们经常被混淆或误用,导致测试失败或注入错误。 本文将从本质区别、使用场景、示例代码、对比表格等多个维度,全面解析这几者的使用方法与差异,助你写出结…...
Spring Boot 项目里设置默认国区时区,Jave中Date时区配置
在 Spring Boot 项目里设置国区时区(也就是中国标准时间,即 Asia/Shanghai),可通过以下几种方式实现: 方式一:在application.properties或application.yml里设置 application.properties properties sp…...
如何使用Cloud Sync搭建群晖NAS跨设备自动备份与手机端实时监控
文章目录 前言1. 群晖NAS安装Cloud Sync1.1 安装和配置 Cloud Sync 2. 功能演示3. 群晖安装cpolar内网穿透3.1 配置Cloud Sync公网访问地址 4. 配置固定公网地址 前言 在当今这个信息爆炸的时代,个人和企业都面临着一个共同的难题:数据分散在各种设备和…...
git回退到指定版本
查看提交历史 在本地仓库中运行 git log 命令,找到需要回退到的版本对应的 commit ID。 重置本地分支 执行 git reset --hard commit_id 将本地分支回退至目标版本,commit_id为要会退的版本号。 强制推送更改 运行 git push origin HEAD --force 强制更…...
【Python爬虫】简单介绍2
目录 四、网页数据的加载方式 4.1 数据直接放在请求响应结果中 4.2 数据在异步请求响应结果中 五、爬虫 - 反爬虫 - 反反爬虫 5.1 反爬虫 5.2 反反爬虫 六、风险 6.1 合法性与合规性的重要性 6.2 违规使用爬虫面临的法律风险示例 6.3 合法合规使用爬虫的建议 四、网页…...
基于QtC++音乐播放器whisper语音转文字歌词解析
演示视频 github源码地址 gitee源码地址 通过网盘分享的文件:MySoftWare 链接: https://pan.baidu.com/s/1JdtZNoMcv7jXR_ELMuqZEg?pwd1yw6 提取码: 1yw6 –来自百度网盘超级会员v4的分享...
【QT】QT界面的美容院 -- QSS
一、背景介绍 🔥 在网页前端开发领域中,CSS 是一个至关重要的部分,描述了一个网页的 “样式”,从而起到对网页 美化 的作用。 所谓 样式 ,包括不限于大小、位置、颜色、背景、间距、字体等等。现在的网页很难找到没有…...
【AI】使用Huggingface模型实现文本内容摘要器
【AI】使用Huggingface模型实现文本内容摘要器 推荐超级课程: 本地离线DeepSeek AI方案部署实战教程【完全版】Docker快速入门到精通Kubernetes入门到大师通关课AWS云服务快速入门实战目录 【AI】使用Huggingface模型实现文本内容摘要器什么是摘要器?摘要器的应用场景什么是…...
centOS 安装和配置docker
以下是在 CentOS 系统上安装和配置 Docker 的详细步骤: 一、安装 Docker 1. 卸载旧版本(如有) sudo yum remove docker \docker-client \docker-client-latest \docker-common \docker-latest \docker-latest-logrotate \docker-logrotate …...
现在AI大模型能帮做数据分析吗?
实际上,有了AI大模型的支持才使得AI数据分析的功能得到更好地应用,比如小浣熊AI支持数据交互功能,只要你输入具体的数据分析要求,它就能自动帮你完成数据清洗、分析、可视化全流程,而且还能生成数据分析报告。 当然&a…...
qt中,父类中有Q_OBJECT,子类中还需要加Q_OBJECT吗
在 Qt 中,关于子类是否需要添加 Q_OBJECT 宏的问题,可以总结如下: 1. 需要添加 Q_OBJECT 的情况 如果子类满足以下任一条件,必须显式添加 Q_OBJECT 宏: 定义了新的信号或槽:即使父类已有 Q_OBJECT&#…...
vue2实现在屏幕中有一个小机器人可以随意移动
第一步:创建store目录结构 src/ ├── store/ │ ├── modules/ │ │ └── robot.js # 机器人专用状态模块 │ └── index.js # Vuex 主配置文件第二步:创建机器人状态模块 创建 src/store/modules/robot.js 文件ÿ…...
MCP协议实战指南:在VS Code中实现PostgreSQL到Excel的自动化迁移
作者:后端小肥肠 🍊 有疑问可私信或评论区联系我。 🥑 创作不易未经允许严禁转载。 姊妹篇: 从PDF到精准答案:Coze助力RAGFlow框架提升数据召回率_提升ragflow-CSDN博客 CozeTreeMind实测:秒出ISO标准流程图…...
Before After:SQL整容级优化
首先说明这个优化有一定提升,但不是我所期望的 我接到一个涉及优化的SQL,具体内容实在太长。而且可能也不利于阅读。于是我脱敏以及简化一下。SQL中间大量的充斥着 (select 列名1 from t1 where t1.id t2.id ) A, (select 列名2 from t1 where t1.id …...
hash.
Redis 自身就是键值对结构 Redis 自身的键值对结构就是通过 哈希 的方式来组织的 哈希类型中的映射关系通常称为 field-value,用于区分 Redis 整体的键值对(key-value), 注意这里的 value 是指 field 对应的值,不是键…...
JMeter重要的是什么
重要特性 支持多种协议: JMeter支持对多种协议进行性能测试,包括HTTP、HTTPS、FTP、JDBC(数据库)、LDAP、JMS、SOAP、REST等。这使得它能够适应各种不同的测试场景。强大的负载模拟能力: JMeter能够模拟大量的虚拟用户…...
Java研学-Activiti7工作流(二)
三 Activiti7 搭建环境 1 开发环境 Jdk1.8或以上版本;Mysql 5及以上的版本;Tomcat8.5;IDEA;Activiti 7.0.0.SR1(流程设计器) 2 安装Activiti流程设计器 ① 在线安装:在Plugins中搜索actiBPM,然后点击Sear…...
鸿蒙开发-编译器使用
15.2编译器使用-目录详解 15.3 编辑器使用-切换工程模块 15.3 编辑器使用-多设备预览 15.1 编辑器使用-编辑中英文...
HTML5+CSS前端开发【保姆级教学】+图像标签附路径问题
引入: Hello!,各位编程猿们!我们知道在网页文档中合理地加入图像,会使文档变得更加生动活泼和引人入胜,而且看上去更加专业、更具有信息性且易于浏览。本期主要介绍图像标签以及超链接标签 一、设置图像标…...
快速启动 Rust + WebAssembly 项目
一、 模板一:wasm-pack-template 适合目标:构建一个 Rust 写的 WebAssembly npm 包 这是最常用、也是最官方推荐的起点模板。它提供了: Cargo.toml 配置好 WebAssembly 的 crate 类型(cdylib)已设置 wee_alloc 和 pan…...
Linux命令-vim编辑
用vi或vim命令进入vim编辑器。 基础: u -- 撤销上一次操作。 x -- 剪切当前光标所在处的字符。 yy -- 复制当前行。 dd -- 剪切当前行。 p -- 粘贴剪贴板内容到光标下方。 i -- 切换到输入模式,在光标当前位置开始输入文本。 :wq -- 保存并退出Vim 编辑器。…...
Windows单机模拟MySQL主从复制
这里写自定义目录标题 下载MySQL ZIP压缩包安装主库1、创建配置文件2、安装服务3、初始化数据库4、启动服务5、配置主库 安装从库1、配置ini文件2、安装服务3、初始化数据库4、启动服务5、配置从库6、验证从库状态 操作主库验证 下载MySQL ZIP压缩包 https://dev.mysql.com/do…...
【区块链+ 人才服务】广州理工学院区块链教学平台 | FISCO BCOS 应用案例
深圳市火链文化传播有限公司与广州理工学院合作,共同建设了“区块链教学平台”。该平台旨在探索区块链技术在教 育领域的应用,特别是在混合式教学模式中的创新实践。 “区块链教学平台”运用了区块链技术的核心优势,包括分布式、信息防篡改以…...
Java Stream深度解析 高阶技巧与性能优化实战
文章目录 一、Stream底层机制揭秘1.1 Stream流水线架构1.2 Spliterator探秘 二、自定义收集器高级实现2.1 实现高性能统计收集器2.2 多级分组优化技巧 三、并行流深度优化3.1 并行度控制策略3.2 工作窃取(Work-Stealing)优化 四、无限流与短路操作4.1 生成无限质数流4.2 短路操…...
【JavaEE初阶】多线程重点知识以及常考的面试题-多线程进阶(一)
本篇博客给大家带来的是多线程的知识点, . 🐎文章专栏: JavaEE初阶 🚀若有问题 评论区见 ❤ 欢迎大家点赞 评论 收藏 分享 如果你不知道分享给谁,那就分享给薯条. 你们的支持是我不断创作的动力 . 王子,公主请阅🚀 要开心要快乐顺便进步 1. 常…...
计算机视觉图像分割入门:阈值、区域生长与分水岭算法
计算机视觉图像分割入门:阈值、区域生长与分水岭算法 一、前言二、图像分割基础概念大揭秘2.1 图像分割的定义2.2 图像分割的目的与作用2.3 图像分割的应用领域三、阈值算法:最简单的图像分割法3.1 原理剖析3.2 手动阈值分割及代码示例3.3 自动阈值分割方法及代…...
android弱网环境数据丢失解决方案(3万字长文)
在移动互联网时代,Android 应用已经成为人们日常生活中不可或缺的一部分。从社交媒体到在线购物,从移动办公到娱乐游戏,用户对应用的依赖程度与日俱增。然而,尽管网络基础设施在全球范围内得到了显著改善,弱网环境依然…...
设计模式:迪米特法则 - 最少依赖,实现高内聚低耦合
一、迪米特法则简介 迪米特法则(Law of Demeter,简称 LoD),也称为“最少知识法则”,核心思想是:一个对象应当对其他对象有最少的了解,仅与直接相关的对象交互。通过减少对象之间的耦合度&#…...
React 把一系列 state 更新加入队列
把一系列 state 更新加入队列 设置组件 state 会把一次重新渲染加入队列。但有时你可能会希望在下次渲染加入队列之前对 state 的值执行多次操作。为此,了解 React 如何批量更新 state 会很有帮助。 开发环境:Reacttsantd 学习内容 什么是“批处理”以…...
AntVG2可视化学习与开发笔记-React19(持续更新)
目录 开始工作 第一步:创建画布空间 第二步:获取画布空间并挂载AntVG2 第三步:进行画布设计配置与数据挂载 第四步:完整代码 实际效果如下 参数理解 一、scale 1. 归一化range:[0,1] 2.nice、domainMin 开始工作 第一…...
从PPT到DeepSeek开启信息可视化的全新之旅
在当今信息爆炸的时代,如何高效、生动地展示信息成为了个人与企业在沟通、汇报、推广等场景中面临的关键挑战。传统的演示工具,如PPT,虽然曾经是展示信息的主力军,但随着技术的发展和人们审美、交互需求的提升,其局限性…...
spark-sql学习内容总结
SparkSession 定义与功能:SparkSession是Spark SQL的入口,封装SparkContext,提供了创建DataFrame和执行SQL的能力。它实质上是SQLContext和HiveContext的组合,因此兼容这两者的API。 创建方式:在使用spark-shell…...
Spring-AI-alibaba 结构化输出
1、将模型响应转换为 ActorsFilms 对象实例: ActorsFilms package com.alibaba.cloud.ai.example.chat.openai.entity;import java.util.List;public record ActorsFilms(String actor, List<String> movies) { } GetMapping("/toBean")public Ac…...
ffmpeg实现视频转码
ffmpeg 实现视频转码 什么是视频编码 视频上传成功后需要对视频进行转码处理。 什么是视频编码? 查阅百度百科如下: 所谓视频编码方式就是指通过压缩技术,将原始视频格式的文件转换成另一种视频格式文件的方式。视频流传输中最为重要的编解…...
【Java学习笔记】Java初级阶段代码规范
Java 初级阶段代码规范 1. 类、方法的注释,要以 javadoc 的方式来写。 2. 非 Java Doc 的注释,往往是给代码的维护者看的,着重告读者为什么这样写,如何修改,注重什么问题等 3. 使用 tab 操作,实现缩进&am…...
适应 AI 时代的软件开发流程:用 AI + TDD 构建可维护项目
🧠 适应 AI 时代的软件开发流程:用 AI + TDD 构建可维护项目 本文面向有系统开发经验的工程师,分享如何结合 Git 管理、AI 协作、YAML 驱动与 TDD 开发方式,高效构建一个可维护、可协作、可交付的嵌入式或通用工程项目。适合 BLE 模块、协议栈组件、物联网控制系统等项目落…...
EasyCVR视频汇聚系统:AIoT+视频智能分析赋能食品安全生产全流程监管
近年来,随着食品安全问题频发,消费者对食品加工企业的信任度逐渐下降,企业生产监管难度加大,市场监管也面临诸多挑战。在这样的背景下,食品加工企业迫切需要通过智能化手段提升生产管理水平,满足消费者和监管部门的要求,同时实现自身业绩的提升。 本文将结合EasyCVR与智…...
ASP.NET Core 性能优化:分布式缓存
文章目录 前言一、分布式缓存的核心概念作用:与内存缓存的区别: 二、ASP.NET Core 中的 IDistributedCache三、常用分布式缓存实现1)Redis(最常用)2)SQL Server3)NCache(企业级方案&…...
一款安全好用的企业即时通讯平台,支持统一门户
在数字化转型的浪潮中,企业面临着信息孤岛、系统分散、协作低效等诸多挑战。BeeWorks作为一款专为企业打造的数字化底座平台,凭借其强大的企业内部应用集成能力和单点登录功能,正在成为企业数字化转型的有力推手。 数字化底座平台࿱…...
C语言 栈 的 描述 和 详解
什么是栈? 栈是一种特殊的线性数据结构。 定义及特点 - 栈是一种只能在一端进行插入和删除操作的特殊线性表。它按照后进先出(Last In First Out,LIFO)的原则存储数据,就像一个只能从顶部取放物品的箱子,…...
.NET MCP 示例
服务器端示例 基础服务器 以下是一个基础的 MCP 服务器示例,它使用标准输入输出(stdio)作为传输方式,并实现了一个简单的回显工具: using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.H…...
【论文阅读】MOE奠基论文《Adaptive Mixtures of Local Experts》
《Adaptive Mixtures of Local Experts》 前言一、让协同学习竞争1.1 方案1.2 方案演变的由来 二、让竞争学习协同2.1 竞争学习2.2 竞争学习协同 三、案例验证3.1 任务背景3.2 实验结果3.3 后续工作 (Future Work) 前言 论文提出了一个基于多个分离网络的有监督学习方案,该方案…...
MATLAB中replace函数用法
目录 语法 说明 示例 替换字符串数组中的子字符串 替换匹配模式的子字符串 替换多个子字符串 replace函数的功能是查找并替换一个或多个子字符串。 语法 newStr replace(str,old,new) 说明 newStr replace(str,old,new) 将所有出现的子字符串 old 替换为 new。如果 …...
MATLAB基本数据类型
1. 数值类型 整数类型: 有符号整数(如 int8, int16, int32, int64)和无符号整数(如 uint8, uint16, uint32, uint64)。 这些类型分别占用 1、2、4、8 个字节,表示不同范围的整数值。 浮点数类…...