【C】线程控制
创建线程
#include <pthread.h>int pthread_create(pthread_t * thread,const pthread_attr_t * attr,void *(*start_routine)(void*), void * arg);
返回值:成功返回0,失败返回错误号。
thread:成功返回后,新创建的线程的id被填写到 thread参数所指向的内存单元。线程id的类型是pthread_t,它只在当前进程中保证是唯一的,在不同的系统中pthread_t这个类型有不同的实现,它可能是一个整数值,也可能是一个结构体,也可能是一个地址,所以不能简单地当成整数用printf打印,调用pthread_self()可以获取当前线程地id。
attr:表示线程属性,这里暂不讨论。
start_routine和arg:新线程所指向的代码由函数指针start_routine决定。start_routine函数接收一个参数arg,该参数的类型为void *,这个指针按什么类型解释由调用者自己定义,start_routine的返回值类型也是void *,这个指针的函数的含义同样由调用者自己定义。
#include <pthread.h>
#include <stdio.h>
#include <stdint.h>
#include <unistd.h>static void thread_func(const char *arg)
{printf("arg=%s\n", arg);pthread_t tid = pthread_self();printf("new thread pid=%u\n", (uint32_t)tid);
}int main(void)
{pthread_t id;int ret = pthread_create(&id, NULL, thread_func, "i am iron man");if (0 != ret){printf("pthread_carete error=%s", strerror(ret));}printf("new thread pid=%u\n", (uint32_t)id);while (1){usleep(1000);}return 0;
}
输出结果为:
new thread pid=1
arg=i am iron man
new thread pid=1
可知,在window上,thread_t类型是一个整型值。
终止线程
如果需要终止某个线程而不终止整个进程,可以有三种方法:
- 从线程函数
return。这种方法对主线程不适用,从main函数return相当于调用exit。 - 一个线程可以调用
pthread_cancel终止同一进程中地另一个线程。 - 线程可以调用
pthread_exit终止自己。
在介绍终止之前,我们先介绍一下pthread_join函数。
#include <pthread.h>int pthread_join(pthread_t thread, void **value_ptr);//返回值:成功返回0,失败返回错误号。
调用该函数地线程将挂起等待,直到id为thread的线程终止。thread线程以不同的方法终止,通过pthread_join得到的终止状态时不同的,总结如下:
- 如果
thread线程通过return返回,value_ptr所指向的单元里存放的是thread线程函数的返回值。 - 如果
thread线程被别的线程调用thread_cancel异常终止掉,value_ptr所指向的单元里存放的是常数PTHREAD_CANCELED。 - 如果
thread线程是自己调用pthread_exit终止的,value_ptr所指向的单元里存放的是传给pthread_exit的参数。
如果对thread线程的终止状态不感兴趣,可以传NULL给value_ptr参数。
一般情况下,线程终止后,其终止状态一直保留到其他线程调用pthread_join获取它的状态为止。但是线程也可以被设置为detach状态,这样的线程一旦终止就立刻回收它占用的所有资源,而不保留终止状态。不能对一个已经处于detach状态的线程调用pthread_join,这样的调用将返回EINVAL。对一个尚未detach的线程调用pthread_join或pthread_detach都可以把该线程设置为detach状态,也就是说,不能对同一线程调用两次pthread_join,或者如果已经对一个线程调用了pthread_detach就不能再调用pthread_join了。
#include <pthread.h>int pthread_detach(pthread_t tid);//返回值:成功返回0,失败返回错误号。
#include <pthread.h>
#include <stdio.h>
#include <stdint.h>
#include <unistd.h>static void *thread1_func(const char *arg)
{printf("arg=%s\n", arg);pthread_t tid = pthread_self();printf("new thread pid=%u\n", (uint32_t)tid);return (void *)0xFF;
}static void *thread2_func(const char *arg)
{printf("arg=%s\n", arg);pthread_t tid = pthread_self();printf("new thread pid=%u\n", (uint32_t)tid);pthread_exit((void *)0xFE);
}static void *thread3_func(const char *arg)
{printf("arg=%s\n", arg);pthread_t tid = pthread_self();printf("new thread pid=%u\n", (uint32_t)tid);while (1){printf("%s running\n", arg);usleep(5000 * 100);}
}int main(void)
{pthread_t id1;int ret1 = pthread_create(&id1, NULL, thread1_func, "i am thread1");if (0 != ret1){printf("pthread_carete error=%s", strerror(ret1));}void *thread1_ret;pthread_join(id1, &thread1_ret);printf("thread1_ret=%d\n", (int)thread1_ret);pthread_t id2;int ret2 = pthread_create(&id2, NULL, thread2_func, "i am thread2");if (0 != ret2){printf("pthread_carete error=%s", strerror(ret2));}void *thread2_ret;pthread_join(id2, &thread2_ret);printf("thread2_ret=%d\n", (int)thread2_ret);pthread_t id3;int ret3 = pthread_create(&id3, NULL, thread3_func, "i am thread3");if (0 != ret3){printf("pthread_carete error=%s", strerror(ret3));}void *thread3_ret;sleep(1);pthread_cancel(id3);pthread_join(id3, &thread3_ret);printf("thread3_ret=%d\n", (int)thread3_ret);while (1){usleep(1000);}return 0;
}
输出为:
arg=i am thread1
new thread pid=1
thread1_ret=255
arg=i am thread2
new thread pid=3
thread2_ret=254
arg=i am thread3
new thread pid=4
i am thread3 running
i am thread3 running
i am thread3 running
i am thread3 running
i am thread3 running
i am thread3 running
i am thread3 running
i am thread3 running
i am thread3 running
i am thread3 running
i am thread3 running
i am thread3 running
i am thread3 running
i am thread3 running
i am thread3 running
i am thread3 running
i am thread3 running
i am thread3 running
i am thread3 running
i am thread3 running
我们可以看到,线程1和线程2的终止与我们的预期一样,但是线程3使用pthread_cancel()似乎没有真正终止线程。那这里就需要引用另外两个概念了。
线程终止状态
@state:PTHREAD_CANCEL_ENABLE:线程对cancel信号立即有反应,将设置为CANCEL状态 (默认)PTHREAD_CANCEL_DISABLE:如果线程state为不可取消,线程不理会信号,继续执行,而使用cancel函数的线程会一直阻塞到可取消状态@oldstate:NULL:state写入有效,即当前只想设置属性,而不关心原来的属性不为NULL:state不写入,保持原有的设定,且获取原来的属性return:成功返回0,错误返回errno EINVAL:无效的stateint pthread_setcancelstate(int state,int *oldstate)
线程终止类型
@type:PTHREAD_CANCEL_DEFERRED:运行到下一个取消点就退出 (默认)PTHREAD_CANCEL_ASYNCHRONOUS:直接退出@oldtyoe:NULL:type写入有效,即当前只想设置属性,而不关心原来的属性不为NULL:type不写入,保持原有的设定,且获取原来的属性return:成功返回0,错误返回errno EINVAL:无效的type
int pthread_setcanceltype(int type,int *oldtype)
所以,如下想要使用pthread_cancel终止线程,那么我们必须在线程函数中调用
pthread_setcancelstate(PTHREAD_CANCEL_ENABLE, NULL);
pthread_setcanceltype(PTHREAD_CANCEL_ASYNCHRONOUS, NULL);
当我们把线程函数改成:
static void *thread3_func(const char *arg)
{pthread_setcancelstate(PTHREAD_CANCEL_ENABLE, NULL);pthread_setcanceltype(PTHREAD_CANCEL_ASYNCHRONOUS, NULL);printf("arg=%s\n", arg);pthread_t tid = pthread_self();printf("new thread pid=%u\n", (uint32_t)tid);while (1){printf("%s running\n", arg);usleep(5000 * 100);}
}
那么就可以正常终止了,输出结果如下:
arg=i am thread1
new thread pid=1
thread1_ret=255
arg=i am thread2
new thread pid=3
thread2_ret=254
arg=i am thread3
new thread pid=4
i am thread3 running
i am thread3 running
thread3_ret=-559038737
相关文章:
【C】线程控制
创建线程 #include <pthread.h>int pthread_create(pthread_t * thread,const pthread_attr_t * attr,void *(*start_routine)(void*), void * arg);返回值:成功返回0,失败返回错误号。 thread:成功返回后,新创建的线程的…...
Maven工程打jar包的N种方式
Maven工程打jar包 一、IDEA自带打包插件二、maven插件打包2.1 制作瘦包(直接打包,不打包依赖包)2.2 制作瘦包和依赖包(相互分离)2.3 制作胖包(项目依赖包和项目打为一个包)2.4 制作胖包…...
一文了解GPU并行计算CUDA
了解GPU并行计算CUDA一、CUDA和GPU简介二、GPU工作原理与结构2.1、基础GPU架构2.2、GPU编程模型2.3、软件和硬件的对应关系三、GPU应用领域四、GPUCPU异构计算五、MPI与CUDA的区别一、CUDA和GPU简介 CUDA(Compute Unified Device Architecture)…...
全网资料最全Java数据结构与算法(1)
一、数据结构和算法概述 1.1什么是数据结构? 官方解释: 数据结构是一门研究非数值计算的程序设计问题中的操作对象,以及他们之间的关系和操作等相关问题的学科。 大白话: 数据结构就是把数据元素按照一定的关系组织起来的集合&a…...
【项目实战】SpringMVC拦截器HandlerInterceptor入门介绍
一、拦截器介绍 拦截器是应用程序级框架中常用的拦截用户请求、实施业务流程控制的模式,它可以将一些公共的、重复发生的业务逻辑从业务处理代码中独立出来,使系统的结构更加清晰,程序的复杂度也减小了。 拦截器是一个常见的特性,它可以实现任何自定义功能,而无需调整业…...
阿里淘宝新势力造型合伙人P8、年薪百万的欧阳娜娜也躲不过的魔鬼面试,看的我心服口服
阿里淘宝新势力造型合伙人P8、年薪百万的欧阳娜娜跳槽了,这不是关键。 她参加了网易有道明星语音录音员/代言人的面试,这也不是关键。 关键是她教科书式的面试过程,狠狠地给我们上了一课。 我是无意间刷到的这个视频的时候,就一…...
深度学习笔记:不同的反向传播迭代方法
1 随机梯度下降法SGD 随机梯度下降法每次迭代取梯度下降最大的方向更新。这一方法实现简单,但是在很多函数中,梯度下降的方向不一定指向函数最低点,这使得梯度下降呈现“之”字形,其效率较低 class SGD:"""随机…...
ElasticSearch 学习笔记总结(三)
文章目录一、ES 相关名词 专业介绍二、ES 系统架构三、ES 创建分片副本 和 elasticsearch-head插件四、ES 故障转移五、ES 应对故障六、ES 路由计算 和 分片控制七、ES集群 数据写流程八、ES集群 数据读流程九、ES集群 更新流程 和 批量操作十、ES 相关重要 概念 和 名词十一、…...
深入理解border以及应用
深入border属性以及应用👏👏 border这个属性在开发过程中很常用,常常用它来作为边界的。但是大家真的了解border吗?以及它的形状是什么样子的。 我们先来看这样一段代码:👏 <!--* Author: syk 185901…...
如何复现论文?什么是论文复现?
参考资料: 学习篇—顶会Paper复现方法 - 知乎 如何读论文?复现代码?_复现代码是什么意思 - CSDN 我是如何复现我人生的第一篇论文的 - 知乎 在我看来,论文复现应该有一个大前提和分为两个层次。 大前提是你要清楚地懂得自己要…...
22.2.28打卡 Codeforces Round #851 (Div. 2) A~C
A题 One and Two 题面翻译 题目描述 给你一个数列 a1,a2,…,ana_1, a_2, \ldots, a_na1,a2,…,an . 数列中的每一个数的值要么是 111 要么是 222 . 找到一个最小的正整数 kkk,使之满足: 1≤k≤n−11 \leq k \leq n-11≤k≤n−1 , anda1⋅a2⋅……...
Learining C++ No.12【vector】
引言: 北京时间:2023/2/27/11:42,高数考试还在进行中,我充分意识到了学校的不高级,因为题目真的没什么意思,虽然挺平易近人,但是……,考试期间时间比较放松,所以不能耽误…...
【数电基础】——逻辑代数运算
目录 1.概念 1.基本逻辑概念 2.基本逻辑电路(与或非) 逻辑与运算 与门电路: 逻辑或运算 或门电路: 逻辑非运算(逻辑反) 非门电路编辑 3.复合逻辑电路(运算) 与非逻辑…...
【Redis】什么是缓存与数据库双写不一致?怎么解决?
1. 热点缓存重建 我们以热点缓存 key 重建来一步步引出什么是缓存与数据库双写不一致,及其解决办法。 1.1 什么是热点缓存重建 在实际开发中,开发人员使用 “缓存 过期时间” 的策略来实现加速数据读写和内存使用率,这种策略能满足大多数…...
互联网衰退期,测试工程师35岁之路怎么走...
国内的互联网行业发展较快,所以造成了技术研发类员工工作强度比较大,同时技术的快速更新又需要员工不断的学习新的技术。因此淘汰率也比较高,超过35岁的基层研发类员工,往往因为家庭原因、身体原因,比较难以跟得上工作…...
动态规划(以背包问题为例)
1) 要求达到的目标为装入的背包的总价值最大,并且重量不超出2) 要求装入的物品不能重复动态规划(Dynamic Programming)算法的核心思想是:将大问题划分为小问题进行解决,从而一步步获取最优解的处理算法。动态规划算法与分治算法类似ÿ…...
Java异常
异常的体系结构 在java的Throwable下有Error和Exception两个子类 Error(错误):程序运行中出现了严重的问题,非代码性错误,无法处理,常见的有虚拟机运行错误和内存溢出等Exception(异常):是由于代码本身造成的问题,可以进行处理,异常一个可以分为运行时异常和编译时异常 运行…...
别克GL8改装完工,一起来看看效果
①豪华商务头等舱 别克GL8作为商务车,不管是家用还是商务接待,原车内饰都太掉档次了,所以车主要求全部换掉。>>织布座椅换成航空座椅 主副驾:改装纳帕皮 中排:改装水晶宝座豪华版航空座椅,带通风、加…...
mac 中 shell 一些知识
mac 设置环境变量首先得看你所使用的 shell shell 是一个命令行解释器,顾名思义就是机器外面的一层壳,用于人机交互,只要是人与电脑之间交互的接口,就可以称为 shell。表现为其作用是用户输入一条命令,shell 就立即解…...
CentOS 配置FTP(开启VSFTPD服务)
网上已经有很多关于VSFTPD的配置,但有两个通病,要么就是原理介绍太多,要么就是不完整,操作下来又要查询多篇文章才能用。 我这里不讲原理,只记录操作,尽可能通过复制下面的操作可以实现FTP读写功能。方便自…...
回形取数-进阶题5
回形取数 题目 问题描述 回形取数就是沿矩阵的边取数,若当前方向上无数可取或已经取过,则左转90度。一开始位于矩阵左上角,方向向下。输入说明 输入第一行是两个不超过200的正整数m, n,表示矩阵的行和列。接下来m行每行n个整数&am…...
SPH与Lagrange混合建模在超高速碰撞仿真中的应用——基于Ls-Dyna的实践探索
1. 为什么需要混合建模?聊聊超高速碰撞仿真的“老大难” 大家好,我是老张,在CAE仿真这个行当里摸爬滚打了十几年,尤其跟Ls-Dyna打交道的时间最长。今天想和大家深入聊聊一个在超高速碰撞仿真中特别实用,但也让很多新手…...
立创天空星ODrive扩展板:双路无刷电机驱动与SimpleFOC/ODrive框架实战
立创天空星ODrive扩展板:双路无刷电机驱动与SimpleFOC/ODrive框架实战 最近在做一个机器人关节项目,需要同时精确控制两个无刷电机,既要力矩平稳,又要位置准确。市面上现成的驱动板要么太贵,要么功能单一,于…...
Pympress实战:用双屏模式录制带备注的演示视频(含常见问题解决方案)
Pympress实战:用双屏模式录制带备注的演示视频(含常见问题解决方案) 你是否曾为录制一场技术分享或在线课程而手忙脚乱?既要关注幻灯片内容,又要顾及演讲节奏,还得确保录屏画面清晰、声音干净。传统的录屏方…...
阴阳师智能高效挂机解决方案:让游戏收益自动化的全新体验
阴阳师智能高效挂机解决方案:让游戏收益自动化的全新体验 【免费下载链接】yysScript 阴阳师脚本 支持御魂副本 双开 项目地址: https://gitcode.com/gh_mirrors/yy/yysScript 作为阴阳师玩家,你是否也曾经历过这样的困境:每天花费数小…...
从零到手搓一个Agent:AI Agents新手入门精通
这一天,你的女朋友问你(假设我们有女朋友),宝宝,什么是Agent啊,Agent和LLM有什么区别呀,最近大家都在说的Agent究竟是什么,包括很多文章都在写的Agent,还有之前谷歌发布的…...
智慧教育+虚拟仿真:解锁煤矿专业实训新范式
煤炭作为我国主体能源,行业安全生产与智能化转型对高素质技术技能人才提出迫切需求,职业院校煤矿专业迎来发展新机遇的同时,也面临着实训教学与行业需求精准对接的挑战。煤矿安全意识警示教育与防治实训室的建设,正是以新一代信息…...
用Cython将UltraFrame再提速10倍:手写C扩展实战
摘要:上一篇UltraFrame已实现18倍性能提升,但聚合运算仍受Python GIL限制。本文手写Cython扩展,绕过GIL实现真并行,聚合速度再提10倍,单机突破亿行/秒处理速度。含完整Cython源码、编译配置与性能调优秘籍。 下期预告:《GPU加速UltraFrame:用CUDA实现千亿行实时分析》 …...
ProcessHacker线程死锁检测:诊断应用程序无响应的原因
ProcessHacker线程死锁检测:诊断应用程序无响应的原因 【免费下载链接】systeminformer A free, powerful, multi-purpose tool that helps you monitor system resources, debug software and detect malware. Brought to you by Winsider Seminars & Solution…...
构建无限免费的AI编程伙伴:VSCode + Roo Code + Gemini Balance负载均衡策略详解
1. 为什么你需要一个“无限免费”的AI编程伙伴? 作为一名写了十几年代码的老兵,我太懂那种感觉了:面对一个棘手的技术难题,或者想快速搭建一个功能模块时,多么希望身边能有一个随时待命、知识渊博的“编程大神”可以请…...
