IO进程线程(八)线程
文章目录
- 一、线程(LWP)概念
- 二、线程相关函数
- (一)创建 pthread_create
- 1. 定义
- 2. 使用(不传参)
- 3. 使用(单个参数)
- 4. 使用(多个参数)
- 5. 多线程执行的顺序
- 6. 多线程内存空间
- (二)获取线程号 pthread_self
- (三)退出线程 pthread_exit
- (四)线程 pthread_join
- (五)标记分离态 pthread_detach
- (六)取消线程pthread_cancel
- (七)多线程文件拷贝
- 三、线程的互斥
- (一)定义和初始化
- (二)上锁
- (三)解锁
- (四)销毁
一、线程(LWP)概念
轻量级的进程,进程是资源分配的基本单位,线程是系统调度的基本单位。
线程不会分配内存空间,一个进程中的多个线程是共用进程的内存空间的(0-3G)
多线程没有多进程安全,但是多线程的效率更高。
每个线程都有自己的task_struct结构体,每个线程都独立的参与时间片轮转
多线程的函数是第三方库实现的。
编码时需要加头文件 #include <pthread.h>
编译时 需要链接 线程库 -lpthread
线程之间是相互平等的
二、线程相关函数
(一)创建 pthread_create
1. 定义
#include <pthread.h>int pthread_create(pthread_t *thread, const pthread_attr_t *attr,void *(*start_routine) (void *), void *arg);功能:创建线程参数:thread:线程id tid attr:线程属性 NULL 表示默认属性start_routine:线程体(线程处理函数)arg:给线程处理函数传参的 如果没有参数可传 可以置NULL返回值:成功 0失败 错误码
- 注:
- pthread_t是一个long类型的数
- start_routine是返回值和参数都是void*的函数指针
- 主线程结束,整个进程结束
2. 使用(不传参)
#include <my_head.h>//线程处理函数
void *task_func(void *arg){printf("我是子线程\n");
}int main(int argc, const char *argv[])
{pthread_t tid = 0;int ret = 0;if(0 != (ret = pthread_create(&tid, NULL, task_func, NULL))){printf("pthread_create : %s\n", strerror(ret));exit(-1);}printf("我是主线程\n");return 0;
}
- 注:在不传参时,最后一个参数可以置NULL
3. 使用(单个参数)
4. 使用(多个参数)
5. 多线程执行的顺序
多线程执行也没有先后顺序,也是时间片轮转,上下文切换
6. 多线程内存空间
同一个进程中的多个线程共用进程内存空间
全局区、堆区、字符串常量区都是公用的
只有栈区是每个线程独立的(作用域问题)
线程之间通信简单,使用全局变量即可,但是不安全
(二)获取线程号 pthread_self
#include <pthread.h>pthread_t pthread_self(void);功能:获取调用线程的线程号 tid参数:无返回值:总是会成功 返回tid
ps -eLf 此时显示的线程号经过了优化方便人查看,并非实际的线程号
(三)退出线程 pthread_exit
#include <pthread.h>
void pthread_exit(void *retval);
功能:退出当前的线程
参数:retval:退出线程时返回的状态值(其他线程用pthread_join来接收)
返回值:无
线程退出的四种情况:
- 线程处理函数执行结束
- 线程中调用pthread_exit退出
- 同一进程中的任一线程中调用exit或者主函数return
- 接收到发送的pthread_cancel取消信号时
(四)线程 pthread_join
结合态线程:结束时必须由其他线程(并没有要求必须创建这个线程的线程来调用)调用pthread_join来回收资源,如果结合态的线程结束后没有回收资源,默认属性是结合态。
分离态线程:结束时由操作系统自动回收其资源
#include <pthread.h>int pthread_join(pthread_t thread, void **retval);功能:等待指定的结合态的线程结束 为其回收资源参数:thread:等待结束的线程号retval:线程结束时的退出状态值 如果不关心可以传NULL返回值:成功 0失败 错误码
- 注:
- 该函数是阻塞等待
- 第二个参数一般传NULL,不关心返回值
- 注意不能返回局部变量的地址,可以返回全局变量的地址,或者static修饰的局部变量的地址,或者malloc在堆区分配的地址(但是要注意回收资源的线程要free)
(五)标记分离态 pthread_detach
#include <pthread.h>int pthread_detach(pthread_t thread);功能:标记一个线程为分离态 结束时由操作系统自动回收资源参数:thread:线程号返回值:成功 0失败 错误码
- 注:在主线程和子线程本身中标记子线程为分离态均可以,但是在子线程中标记自己更好,因为线程执行不分先后顺序,在子线程中标记,可以确保在子线程未执行结束前可以标记成功。
- 一般在子线程开头进行标记,当作是线程属性的一种
(六)取消线程pthread_cancel
#include <pthread.h>
int pthread_cancel(pthread_t thread);
功能:给线程发一个取消的请求目标线程是否以及何时响应这个请求取决于线程的两种属性:statu和type
参数:thread:线程id
返回值:成功 0失败 错误码
#include <pthread.h>
int pthread_setcancelstate(int state, int *oldstate);
//设置线程是否可被取消
PTHREAD_CANCEL_ENABLE 可被取消 ----默认属性
PTHREAD_CANCEL_DISABLE 不可被取消
#include <pthread.h>
int pthread_setcanceltype(int type, int *oldtype);
//设置线程的取消类型
PTHREAD_CANCEL_DEFERRED 延时取消 ----默认属性直到线程下一次调用能作为取消点的函数时才会被取消
PTHREAD_CANCEL_ASYNCHRONOUS 立即取消
不可被取消,不接收取消信号
可被取消,立即取消,在接收到取消信号后立即取消;
可被取消,延时取消,在接收到取消信号先将当前命令执行完后,如果再次遇到可作为取消点的函数时被取消;如果遇不到就无法取消
- 注:默认是可被取消,延时取消
(七)多线程文件拷贝
功能需求:
使用多线程拷贝文件
代码实现:
//实现多线程拷贝文件
#include <my_head.h>typedef struct _msg{char *src_file;char *dest_file;int offset;int len;
}msg_t;//初始化:保证有一个清空的目标文件,获取源文件的长度
int init_cp(const char *src_file, const char *dest_file){//打开一个目标文件,不存在就创建,存在就清空FILE *dest_fp=fopen(dest_file,"w");if(NULL == dest_fp) ERR_LOG("open dest file error");fclose(dest_fp);//获取源文件长度FILE *src_fp=fopen(src_file,"r");if(NULL == src_fp) ERR_LOG("open src file error");fseek(src_fp,0,SEEK_END);int size = ftell(src_fp);fclose(src_fp);return size;
}//线程处理函数
void *func_cp(void *argv1){printf("子线程函数:%ld\n",pthread_self());//使用结构体接收参数msg_t t_argv=*(msg_t *)argv1;//打开文件int src_fd = open(t_argv.src_file,O_RDONLY);if(-1 == src_fd) ERR_LOG("open src file error");int dest_fd = open(t_argv.dest_file,O_WRONLY);if(-1 == dest_fd) ERR_LOG("open dest file error");//将两个文件的指针移动到offset位置lseek(src_fd,t_argv.offset,SEEK_SET);lseek(dest_fd,t_argv.offset,SEEK_SET);//复制函数int w_byte=0;//记录写入的字节数int r_byte=0;//记录本次读到的字节数char buff[10];//缓冲区while(0 < (r_byte=read(src_fd,buff,sizeof(buff)))){w_byte+=r_byte;if(w_byte>=t_argv.len){write(dest_fd,buff,t_argv.len-(w_byte-r_byte));break;}write(dest_fd,buff,r_byte);}//自己就结束了
}int main(int argc, char const *argv[])
{if(3 != argc){printf("Usage:%s src dest\n",argv[0]);exit(-1);}//初始化int size = init_cp(argv[1],argv[2]);char src_path[20]={0};char dest_path[20]={0};strcpy(src_path,argv[1]);strcpy(dest_path,argv[2]);//创建线程msg_t argv2={src_path,dest_path,0,size/2};pthread_t tid1=0;int sta=0;if(sta = pthread_create(&tid1,NULL,func_cp,(void *)&argv2)){printf("pthread_create error:%s\n",strerror(sta));}msg_t argv1={src_path,dest_path,size/2,size-size/2};pthread_t tid2=0;if(sta = pthread_create(&tid2,NULL,func_cp,(void *)&argv1)){printf("pthread_create error:%s\n",strerror(sta));}pthread_join(tid1,NULL);pthread_join(tid2,NULL);return 0;
}
进阶版:线程数由命令行传参进来
三、线程的互斥
保证临界资源同一时刻仅被一个线程访问
- 注:
- 上锁范围尽量小,尽量在进程访问临界变量时再上锁
- 只有加锁的线程可以解锁
- 对临界变量加锁,利用的是锁自身机制:一个线程加锁,另一个线程想要加锁会阻塞
- 不要出现死锁:退出线程前,解锁;一次只拿一把锁。
(一)定义和初始化
#include <pthread.h>
//静态初始化
pthread_mutex_t lock;
pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER;//静态初始化
//动态初始化互斥锁
int pthread_mutex_init(pthread_mutex_t *mutex,const pthread_mutexattr_t *mutexattr);功能:动态初始化互斥锁参数:mutex:要初始化的锁mutexattr:属性 一般传 NULL 默认属性返回值:总是会成功 返回0
(二)上锁
int pthread_mutex_lock(pthread_mutex_t *mutex);
功能:上锁(如果没有锁可以上锁,线程将被阻塞)
参数:mutex 互斥锁指针
返回值:成功 0 失败 错误码int pthread_mutex_trylock(pthread_mutex_t *mutex);
功能:尝试上锁(如果没有锁可以上锁,线程不会被阻塞,而是立即返回错误)
参数:mutex 互斥锁指针
返回值:成功 0 失败 错误码
(三)解锁
int pthread_mutex_unlock(pthread_mutex_t *mutex);功能:解锁(一般使用时,只有加锁的线程可以解锁)参数:mutex 互斥锁指针返回值:成功 0 失败 错误码
(四)销毁
int pthread_mutex_destroy(pthread_mutex_t *mutex);功能:在不需要使用互斥锁的时候销毁互斥锁参数:mutex 互斥锁指针返回值:成功 0 失败 错误码
相关文章:
IO进程线程(八)线程
文章目录 一、线程(LWP)概念二、线程相关函数(一)创建 pthread_create1. 定义2. 使用(不传参)3. 使用(单个参数)4. 使用(多个参数)5. 多线程执行的顺序6. 多线程内存空间 ࿰…...
Linux基础指令网络管理003
本章主要讲述如何进行网络诊断。 操作系统: CentOS Stream 9 操作步骤: 操作指令 ping: 测试网络连接的连通性和延迟。 [rootlocalhost ~]# ping 192.168.80.111 PING 192.168.80.111 (192.168.80.111) 56(84) 比特的数据。 64 比特&a…...
在Android中使用 MQTT 服务实现消息通信
1.摘要 MQTT(Message Queuing Telemetry Transport,消息队列遥测传输)是一种轻量级的、基于发布/订阅(Publish/Subscribe)模式的通信协议,最初由 IBM 在1999年开发。它设计用于在低带宽、不稳定的网络环境下…...
qsort函数
学习c语言的过程中少不了的就是排序,例如冒泡排序(不清楚的同学可以翻找一下之前的文章), 我们这里将冒泡排序作为一个自定义函数来呈现一下 #include<stdio.h>void bubble_sort(int arr[], int len) {for (int i 0; i &…...
你可以直接和数据库对话了!DB-GPT 用LLM定义数据库下一代交互方式,数据库领域的GPT、开启数据3.0 时代
✨点击这里✨:🚀原文链接:(更好排版、视频播放、社群交流、最新AI开源项目、AI工具分享都在这个公众号!) 你可以直接和数据库对话了!DB-GPT 用LLM定义数据库下一代交互方式,数据库领…...
数据结构笔记2 栈和队列
为什么在循环队列中,判断队满的条件是(Q.rear1)模maxqsize? 取模运算(%)在循环队列中起到关键作用,主要是因为它能确保索引值在数组的有效范围内循环。具体来说,取模运算有以下几个重要作用&am…...
Python | 刷题笔记
继承 class Father:__secret"you are your own kid"stroy"iam a handsome boy..."def tellstory(self):print("我的故事:",self.stroy)def __tellstory(self):print("我的秘密:",Father.__secret) class Son(Father):def tell(self…...
软件三班20240605
文章目录 1.创建工程和模块2.添加 web支持3.创建前端代码4.添加servlet 依赖5. 代码6.案例2 1.创建工程和模块 2.添加 web支持 方法1 方法2 3.创建前端代码 4.添加servlet 依赖 5. 代码 <!DOCTYPE html> <html lang"en"> <head><meta c…...
http和https数据传输与协议区分
目录 1. 数据传输安全性2. 端口号3. URL 前缀4. SSL/TLS 证书5. 性能6. SEO 和用户信任7. 应用场景总结 HTTP(HyperText Transfer Protocol)和 HTTPS(HyperText Transfer Protocol Secure)是用于在客户端(如浏览器&…...
天才程序员周弈帆 | Stable Diffusion 解读(一):回顾早期工作
本文来源公众号“天才程序员周弈帆”,仅用于学术分享,侵权删,干货满满。 原文链接:Stable Diffusion 解读(一):回顾早期工作 在2022年的这波AI绘画浪潮中,Stable Diffusion无疑是最…...
软件架构初探
MVC架构软件层次结构是面向实体的,他最底层是实体类,实体类中封装了对象的抽象数据类型(数据结构和对数据结构的基本操作)。然后向上一层数据处理层提供接口,数据处理层利用模型层提供的对象和基本操作进一步进行算法的…...
Python01 -分解整包数据到各个变量操作和生成器
Python 的星号表达式可以用来解决这个问题。比如,你在学习一门课程,在学期末的时候,你想统计下家庭作业的平均成绩,但是排除掉第一个和最后一个分数。如果只有四个分数,你可能就直接去简单的手动赋值,但如果…...
flutter image_picker 执行拍照的图片怎么保存到本地
在 Flutter 中,使用 image_picker 插件拍照的图片默认会被保存到设备的临时目录中。这个临时目录的具体位置取决于设备的操作系统。在 iOS 上,它通常是应用的沙盒目录;在 Android 上,它通常是应用的缓存目录。 这些图片不会被自动…...
基于Python的北京天气数据可视化分析
项目用到库 import numpy as np import pandas as pd import datetime from pyecharts.charts import Line from pyecharts.charts import Boxplot from pyecharts.charts import Pie,Grid from pyecharts import options as opts from pyecharts.charts import Calendar 1.2…...
Linux编译器-gcc或g++的使用
一.安装gcc/g 在linux中是不会自带gcc/g的,我们需要编译程序就自己需要安装gcc/g。 很简单我们使用简单的命令安装gcc:sudo yum install -y gcc。 g安装:sudo yum install -y gcc-c。 我们知道Windows上区分文件,都是使用文件…...
一条sql的执行流程
文章地址 https://blog.csdn.net/qq_43618881/article/details/118657040 连接器 请求先走到连接器,与客户端建立连接、获取权限、维持和管理连接 mysql缓存池 如果要查找的数据直接在mysql缓存池里面就直接返回数据 分析器 请求已经建立了连接,现在…...
Android音乐播放器的思路处理
** 1.android音乐播放播放列表中下一首上一首随机播放的思路 ** 实现 Android 音乐播放器的播放列表中的下一首、上一首和随机播放功能涉及到对音乐列表的管理以及对播放顺序的控制。以下是实现这些功能的思路: 下一首和上一首功能: 维护一个音乐列表…...
算法课程笔记——可撤销并查集
算法课程笔记——可撤销并查集 Gv...
【排序算法】快速排序
一、定义: 快速排序是Hoare于1962年提出的一种二叉树结构的交换排序方法(也叫Hoare排序),是一种基于分治的排序方。其基本原理是将待排序的数组通过一趟排序分成两个独立的部分,其中一部分的所有数据比另一部分的所有数…...
OS复习笔记ch7-2
页式管理 学过计组的同学都了解一点页式管理,就是将内存划分成较小的、大小固定的、等大的块。现在OS引入了进程的概念,那么为了匹配内存的分块,同样把进程也划分成同样大小的块。 这里区分两个概念 The chunks of a process are called p…...
4.通用编程概念
目录 一、变量与常量1.1 变量1.2 常量 二、遮蔽三、数据类型3.1 标量类型1. 整型2. 浮点型3. 布尔类型4.字符类型 3.2 复合类型1. 元组2. 数组 四、函数五、语句和表达式六、函数的返回值 一、变量与常量 1.1 变量 在Rust中默认的变量是不可变的,如果修改其值会导致…...
iBeacon赋能AR导航:室内定位技术的原理与优势
室内定位导航对于大型商场、机场、医院等复杂室内环境至关重要,它帮助人们快速找到目的地,提高空间利用率。AR技术通过将虚拟信息叠加在现实世界,提供直观导航指引,正在成为室内导航的新趋势,增强用户互动体验…...
【sklearn】【逻辑回归1】
学习笔记来自: 所用的库和版本大家参考: Python 3.7.1Scikit-learn 0.20.1 Numpy 1.15.4, Pandas 0.23.4, Matplotlib 3.0.2, SciPy 1.1.0 1 概述 1.1 名为“回归”的分类器 在过去的四周中,我们接触了不少带“回归”二字的算法…...
java(kotlin)和 python 通过DoubleCloud的kafka进行线程间通信
进入 DoubleCloud https://www.double.cloud 创建一个kafka 1 选择语言 2 运行curl 的url命令启动一个topic 3 生成对应语言的token 4 复制3中的配置文件到本地,命名为client.properties 5 复制客户端代码 对python和java客户端代码进行了重写,java改成…...
vivado DIAGRAM、HW_AXI
图表 描述 块设计(.bd)是在IP中创建的互连IP核的复杂系统 Vivado设计套件的集成商。Vivado IP集成器可让您创建复杂的 通过实例化和互连Vivado IP目录中的IP进行系统设计。一块 设计是一种分层设计,可以写入磁盘上的文件(.bd&…...
学习分享-为什么把后台的用户验证和认证逻辑放到网关
将后台的用户验证和认证逻辑放到网关(API Gateway)中是一种常见的设计模式,这种做法在微服务架构和现代应用中有许多优势和理由: 1. 集中管理认证和授权 统一的安全策略 在一个包含多个微服务的系统中,如果每个服务…...
27 ssh+scp+nfs+yum进阶
ssh远程管理 ssh是一种安全通道协议,用来实现字符界面的远程登录。远程复制,远程文本传输。 ssh对通信双方的数据进行了加密。 用户名和密码登录 密钥对认证方式(可以实现免密登录) ssh 22 网络层 传输层 数据传输的过程中是…...
LabVIEW液压伺服压力机控制系统与控制频率选择
液压伺服压力机的控制频率是一个重要的参数,它直接影响系统的响应速度、稳定性和控制精度。具体选择的控制频率取决于多种因素,包括系统的动态特性、控制目标、硬件性能以及应用场景。以下是一些常见的指导原则和考量因素: 常见的控制频率范…...
阿里云(域名解析) certbot 证书配置
1、安装 certbot ubuntu 系统: sudo apt install certbot 2、申请certbot 域名证书,如申请二级域名aa.example.com 的ssl证书,同时需要让 bb.aa.example.com 也可以使用此证书 1、命令:sudo certbot certonly -d “域名” -d “…...
Web LLM 攻击技术
概述 在ChatGPT问世以来,我也尝试挖掘过ChatGPT的漏洞,不过仅仅发现过一些小问题:无法显示xml的bug和错误信息泄露,虽然也挖到过一些开源LLM的漏洞,比如前段时间发现的Jan的漏洞,但是不得不说传统漏洞越来…...
南京企业做网站/营销100个引流方案
第二题生日蜡烛(结果填空)某君从某年开始每年都举办一次生日party,并且每次都要吹熄与年龄相同根数的蜡烛。现在算起来,他一共吹熄了236根蜡烛。请问,他从多少岁开始过生日party的?请填写他开始过生日party的年龄数。注意…...
建设一个网站需要做哪些工作内容/网站seo是干什么的
1.1 几种常见存储设备的接口 1.IDE接口 IDE的英文全称为"Integrated Drive Electronics",即"电子集成驱动器",是曾经主流的硬盘接口。IDE接口也称之为ATA接口。ATA的英文拼写为"Advanced Technology Attachment"。2003年推…...
青岛市城乡建设委员会网站电话/百度一下了你就知道官网
TRUNK被解释为“端口汇聚”,是带宽扩展和链路备份的一个重要途径。TRUNK(端口汇聚)功能是将交换机的多个物理端口汇聚在一起形成一个逻辑上的物理端口,同一汇聚组内的多条链路则可视为一条逻辑链路。端口汇聚可以实现用多条链路汇聚成一条逻辑链路增加带…...
北京规划网站/指数搜索
组长安排的工作做完了,闲得蛋疼。重新理解下最经典的冒泡算法。原理:比较相邻的两个数,如果第一个数比后一个要大,则交换位置。其实原理很好理解了,一看就是写个循环。但是要交换几次呢。我们就拿最简单的排序来推理一…...
美国做调查的网站/短视频seo系统
据国外媒体报道,惠普公司8月23日宣布,计划以16亿美元竞购虚拟存储制造商3PAR。而在一周前,戴尔公司曾出价11.5亿美元收购此公司。 惠普是在给3PAR的董事长和CEO的信中透露其收购价格的。惠普执行副总裁兼首席战略和技术官谢恩罗宾逊ÿ…...
域名解析记录值填什么/一个网站可以优化多少关键词
4 案例4:部署Multipath多路径环境 4.1 问题 通过Multipath,实现以下目标: 在共享存储服务器上配置iSCSI,为应用服务器共享存储空间 应用服务器上配置iSCSI,发现远程共享存储 应用服务器上配置Multipath,将相…...