【Linux】线程控制

news2025/1/19 20:44:06

目录

  • 🌈前言
  • 🌸1、Linux线程控制
    • 🍡1.1、创建线程(pthread_create)
    • 🍢1.2、通过指令查看线程PID和LWP
    • 🍧1.3、获取线程ID(pthread_self)
    • 🍨1.4、线程终止(pthread_exit/cancel)
    • 🍨1.5、线程等待(pthread_join)
    • 🍱1.6、分离线程(pthread_detach)

🌈前言

这篇文章给大家带来线程控制的学习!!!


🌸1、Linux线程控制

POSIX线程库(第三方库)
  • 这是Linux自带原生库,定义了创建和操纵线程的一套API(函数接口),绝大多数函数的名字都是以“pthread_”为前缀
  • 要使用这些函数库,要通过引入头文<pthread.h>
  • 链接这些线程函数库时要使用编译器命令的“-lpthread”选项,因为它是第三方库
  • 任何语言在Linux想要实现一套自己的线程库,都是封装POSIX线程库来实现的

🍡1.1、创建线程(pthread_create)

#include <pthread.h>
typedef unsigned long int pthread_t;

int pthread_create(pthread_t *thread, const pthread_attr_t *attr,
					void *(*start_routine) (void *), void *arg);
函数解析
  • 作用:pthread_create()函数在调用过程中启动一个新线程
  • thread:该参数返回一个线程ID输出型参数,由0S填充
  • attr:该参数是用于设置线程的属性,如果为,则设置线程默认的属性,返回值void*可以指定线程退出状态值
  • start_routine:该参数是一个回调函数线程启动后,会执行该函数的代码
  • arg:该参数是用来命名线程的,它会被传给线程启动函数的参数(start_routine
  • 返回值:启动成功返回0,启动失败返回错误码(线程有自己的错误码)

创建一个线程,并且让它跑起来

#include <iostream>
#include <cstdlib>
#include <unistd.h>
#include <pthread.h>
using namespace std;

// |---------------------------------------------------------------------|
// | #include <pthread.h>                                                |
// | int pthread_create(pthread_t *thread, const pthread_attr_t *attr,   |
// |                     void *(*start_routine) (void *), void *arg);    |
// |---------------------------------------------------------------------|

void *CallPthread1(void *arg)
{
    printf("I am %s\n", (const char *)arg);
    // 指定线程退出状态值
    return (void *)0;
}

void *CallPthread2(void *arg)
{
    printf("I am %s\n", (const char *)arg);
    // 指定线程退出状态值
    return (void *)0;
}

int main()
{
    // 1、定义线程id,该id由OS填充
    pthread_t tid1;
    pthread_t tid2;

    // 创建线程,并且填充t1的id,arg参数给线程命名,并且让线程执行CallPthread代码
   if (pthread_create(&tid1, nullptr, CallPthread1, (void *)"thread t1") != 0)
    {
        exit(EXIT_FAILURE);
    }
    
    if (pthread_create(&tid2, nullptr, CallPthread2, (void *)"thread t2") != 0)
    {
        exit(EXIT_FAILURE);
    }
    return 0;
}

运行结果:

[lyh@192 Make_thread(1)]$ ./Thread_Creation 
I am thread t1
I am thread t2

错误检查
  • 传统的一些库函数是:成功返回0,失败返回-1,并且对全局变量errno赋值以指示错误
  • 线程函数出错时,不会设置全局变量errno(但是大部分其他库函数会设置),而是通过函数的返回值错误码返回
  • 线程同样也提供了线程内的errno变量每个线程都有属于自己的局部errno,以避免一个线程干扰另一个线程
  • 对于线程函数的错误,建议通过返回值来判定,因为读取返回值要比读取线程内的errno变量开销更小

🍢1.2、通过指令查看线程PID和LWP

前言
  • 线程是进程的执行程序(执行流),它们的PID是一样的
  • 但是,进程和轻量级进程(线程)ID(LWP)是不一样的
  • 轻量级进程(线程)是CPU调度基本单位,需要一个唯一的数值(LWP)来标识它
  • 我们可以通过ps -aL查看全部正在运行主线程(main函数)新线程(pthread_create)的信息
#include <iostream>
#include <cstdlib>
#include <unistd.h>
#include <pthread.h>
using namespace std;

// |---------------------------------------------------------------------|
// | #include <pthread.h>                                                |
// | int pthread_create(pthread_t *thread, const pthread_attr_t *attr,   |
// |                     void *(*start_routine) (void *), void *arg);    |
// |---------------------------------------------------------------------|

void *CallPthread1(void *arg)
{
    printf("I am %s, thread PID: %d\n", (const char *)arg, getpid());
    sleep(3);
    // 指定线程退出状态值
    return (void *)0;
}

void *CallPthread2(void *arg)
{
    printf("I am %s, thread PID: %d\n", (const char *)arg, getpid());
    sleep(3);
    // 指定线程退出状态值
    return (void *)0;
}

int main()
{
    // 1、定义线程id,该id由OS填充
    pthread_t tid1;
    pthread_t tid2;

    // 创建线程,并且填充t1的id,arg参数给线程命名,并且让线程执行CallPthread代码
    if (pthread_create(&tid1, nullptr, CallPthread1, (void *)"thread t1") != 0)
    {
        exit(EXIT_FAILURE);
    }
    if (pthread_create(&tid2, nullptr, CallPthread2, (void *)"thread t2") != 0)
    {
        exit(EXIT_FAILURE);
    }

    while (true)
    {
        cout << "我是主线程, pid: : " << getpid() << endl;
        sleep(1);
    }
    return 0;
}

在这里插入图片描述


🍧1.3、获取线程ID(pthread_self)

线程ID
  • pthread_ create函数会产生一个线程ID,存放在第一个参数指向的地址中。该线程ID和前面说的线程ID不是一回事
  • 前面讲的线程ID属于进程调度的范畴。因为线程是轻量级进程,是操作系统调度器的最小单位,所以需要一个数值来唯一表示该线程
  • pthread_ create函数第一个参数指向一个虚拟内存单元,该内存单元的地址即为新创建线程的线程ID,属于NPTL线程库的范畴
  • 线程库的后续操作,就是根据该线程ID来操作线程的

线程库NPTL提供了pthread_ self函数,可以获得线程自身的ID:

#include <pthread.h>
typedef unsigned long int pthread_t;

pthread_t pthread_self(void);
函数解析
  • 作用:获取线程自身的ID
  • 返回值:此函数一定会调用成功,返回调用线程的ID
#include <iostream>
#include <cstdlib>
#include <unistd.h>
#include <pthread.h>
using namespace std;

void *CallPthread1(void *arg)
{
    // 获取线程id
    pthread_t id = pthread_self();
    printf("I am %s, pthread PID: %u\n", (const char *)arg, id);
    // 指定线程退出状态值
    return (void *)0;
}

void *CallPthread2(void *arg)
{
    // 获取线程id
    pthread_t id = pthread_self();
    printf("I am %s, pthread PID: %u\n", (const char *)arg, id);
    // 指定线程退出状态值
    return (void *)0;
}

int main()
{
    // 1、定义线程id,该id由OS填充
    pthread_t tid1;
    pthread_t tid2;

    // 创建线程,并且填充t1的id,arg参数给线程命名,并且让线程执行CallPthread代码
    if (pthread_create(&tid1, nullptr, CallPthread1, (void *)"thread t1") != 0)
    {
        exit(EXIT_FAILURE);
    }
    if (pthread_create(&tid2, nullptr, CallPthread2, (void *)"thread t2") != 0)
    {
        exit(EXIT_FAILURE);
    }
    return 0;
}

运行结果:

[lyh@192 Make_thread(1)]$ ./Thread_Creation 
I am thread t1, pthread PID: 3796985600
I am thread t2, pthread PID: 3788592896

🍨1.4、线程终止(pthread_exit/cancel)

如果需要只终止某个线程而不终止整个进程

三种方法
  • 线程函数return。这种方法对主线程不适用,从main函数return相当于调用exit
  • 线程可以调用pthread_ exit函数终止自己
  • 一个线程可以调用pthread_ cancel函数终止同一进程中的另一个线程

pthread_exit函数:

 #include <pthread.h>
typedef unsigned long int pthread_t;

void pthread_exit(void *value_ptr);
函数解析
  • 作用:pthread_exit()函数终止调用线程,就如同进程在结束时调用exit函数一样
  • 返回值:无返回值,此函数一定会调用成功
  • value_ptr:保存线程退出后的返回值,返回一个指向某个对象的指针,该指针不能是线程栈局部变量,后面线程等待可以获取到这个变量
  • 需要注意:pthread_exit或者return返回的指针所指向的内存单元必须是全局的或者是用malloc分配的,不能在线程函数的栈上分配,因为当其它线程得到这个返回指针时线程函数已经退出了
#include <iostream>
#include <cstdlib>
#include <unistd.h>
#include <pthread.h>
using namespace std;

void *CallPthread1(void *arg)
{
    // 获取线程id
    pthread_t id = pthread_self();
    printf("I am %s, pthread PID: %u\n", (const char *)arg, id);
    // 线程终止,线程退出后的值设为空
    pthread_exit(nullptr);
}

void *CallPthread2(void *arg)
{
    // 获取线程id
    pthread_t id = pthread_self();
    printf("I am %s, pthread PID: %u\n", (const char *)arg, id);
    pthread_exit(nullptr);
}

int main()
{
    // 1、定义线程id,该id由OS填充
    pthread_t tid1;
    pthread_t tid2;

    // 创建线程,并且填充t1的id,arg参数给线程命名,并且让线程执行CallPthread代码
    if (pthread_create(&tid1, nullptr, CallPthread1, (void *)"thread t1") != 0)
    {
        exit(EXIT_FAILURE);
    }
    if (pthread_create(&tid2, nullptr, CallPthread2, (void *)"thread t2") != 0)
    {
        exit(EXIT_FAILURE);
    }
    return 0;
}

pthread_cancel函数:

 #include <pthread.h>
typedef unsigned long int pthread_t;

int pthread_cancel(pthread_t thread);
函数解析
  • 作用:pthread_cancel函数向进程中正在运行的线程发送取消请求
  • 返回值:成功返回0,失败返回一个errno错误码
  • thread:线程ID(pthread_create中第一个输出型参数的值)
#include <iostream>
#include <cstdlib>
#include <unistd.h>
#include <pthread.h>
using namespace std;

void *CallPthread1(void *arg)
{
    // 获取线程id
    pthread_t id = pthread_self();
    while (true)
    {
        printf("I am %s, pthread PID: %u\n", (const char *)arg, id);
        sleep(1);
    }
    pthread_exit(nullptr);
}

void *CallPthread2(void *arg)
{
    // 获取线程id
    pthread_t id = pthread_self();
    while (true)
    {
        printf("I am %s, pthread PID: %u\n\n", (const char *)arg, id);
        sleep(1);
    }
    pthread_exit(nullptr);
}

int main()
{
    // 1、定义线程id,该id由OS填充
    pthread_t tid1;
    pthread_t tid2;

    // 创建线程,并且填充t1的id,arg参数给线程命名,并且让线程执行CallPthread代码
    if (pthread_create(&tid1, nullptr, CallPthread1, (void *)"thread t1") != 0)
    {
        exit(EXIT_FAILURE);
    }
    if (pthread_create(&tid2, nullptr, CallPthread2, (void *)"thread t2") != 0)
    {
        exit(EXIT_FAILURE);
    }
    
	// 二秒后退出ID为tid1的线程
    sleep(2);
    pthread_cancel(tid1);
    return 0;
}

运行结果:

在这里插入图片描述


🍨1.5、线程等待(pthread_join)

为什么需要线程等待?
  • 已经退出的线程,其空间没有被释放,仍然在进程的地址空间内
  • 创建新的线程不会复用刚才退出线程的地址空间
 #include <pthread.h>
typedef unsigned long int pthread_t;

int pthread_join(pthread_t thread, void **value_ptr);
函数解析
  • 作用:pthread_join函数等待指定的线程终止,如果该线程已经终止,则pthread_join()立即返回
  • 返回值:成功返回0,失败返回一个errno错误码
  • thread:线程ID(pthread_create中第一个输出型参数的值)
  • value_ptr:它指向一个指针,这个指针(解引用value_ptr)指向线程的返回值
#include <iostream>
#include <cstdlib>
#include <unistd.h>
#include <pthread.h>
using namespace std;

void *CallPthread1(void *arg)
{
    // 获取线程id
    pthread_t id = pthread_self();
    printf("I am %s, pthread PID: %u\n", (const char *)arg, id);
    // p是一个线程状态返回值,传给pthread_exit参数
    const char* p = "线程退出成功1! ! !";
    pthread_exit((void*)p);
}

void *CallPthread2(void *arg)
{
    // 获取线程id
    pthread_t id = pthread_self();
    printf("I am %s, pthread PID: %u\n", (const char *)arg, id);
    const char* p = "线程退出成功2! ! !";
    pthread_exit((void*)p);
}

int main()
{
    // 1、定义线程id,该id由OS填充
    pthread_t tid1;
    pthread_t tid2;

    // 创建线程,并且填充t1的id,arg参数给线程命名,并且让线程执行CallPthread代码
    if (pthread_create(&tid1, nullptr, CallPthread1, (void *)"thread t1") != 0)
    {
        exit(EXIT_FAILURE);
    }
    
    if (pthread_create(&tid2, nullptr, CallPthread2, (void *)"thread t2") != 0)
    {
        exit(EXIT_FAILURE);
    }

    void* RV1;
    void* RV2;
    
    // 等待线程退出,并且获取线程退出时的返回值
    pthread_join(tid1, &RV1);
    pthread_join(tid2, &RV2);
    
    cout << (const char*)RV1 << endl;
    cout << (const char*)RV2 << endl;
    return 0;
}

运行结果:

[lyh@192 Make_thread(1)]$ make
g++ -o Thread_Creation Thread_Creation.cpp -std=c++11 -lpthread

[lyh@192 Make_thread(1)]$ ./Thread_Creation 
I am thread t1, pthread PID: 2257434368
I am thread t2, pthread PID: 2249041664
线程退出成功1! ! !
线程退出成功2! ! !

  • 调用该函数的线程将挂起等待,直到id为thread的线程终止

  • thread线程以不同的方法终止,通pthread_join得到的终止状态是不同的

总结
  • 如果thread线程通过return返回,value_ ptr所指向的内存单元里存放的是thread线程函数的返回值
  • 如果thread线程被别的线程调用pthread_ cancel异常终止,value_ ptr所指向的单元里存放的是v常数PTHREAD_ CANCELED
  • 如果thread线程是自己调用pthread_exit终止的,value_ptr所指向的单元存放的是传给pthread_exit的参数
  • 如果对thread线程的终止状态不感兴趣,可以传NULL给value_ ptr参数

在这里插入图片描述


🍱1.6、分离线程(pthread_detach)

概念
  • 默认情况下,新创建的线程是joinable(连接的)的,线程退出后,需要对其进行pthread_join操作,否则无法释放资源,从而造成系统泄漏
  • 如果不关心线程的返回值join是一种负担,这个时候,我们可以告诉系统,当线程退出时,自动释放线程资源
 #include <pthread.h>
typedef unsigned long int pthread_t;

int pthread_detach(pthread_t thread);
函数解析
  • 作用:pthread_detach函数可以让主线程和线程进行分离,线程自动释放资源
  • 返回值:成功返回0,失败返回一个errno错误码
  • thread:线程ID(pthread_create中第一个输出型参数的值)
注意
  • 可以是线程组内其他线程对目标线程进行分离,也可以是线程自己分离
  • joinable(连接)和分离是冲突的,一个线程不能既是joinable又是分离的

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

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

相关文章

一文读懂PCB阻焊工艺

PCB阻焊油墨根据固化方式&#xff0c;阻焊油墨有感光显影型的油墨&#xff0c;有热固化的热固油墨&#xff0c;还有UV光固化的UV油墨。而根据板材分类&#xff0c;又有PCB硬板阻焊油墨&#xff0c;FPC软板阻焊油墨&#xff0c;还有铝基板阻焊油墨&#xff0c;铝基板油墨也可以用…

力扣算法(Java实现)—数组入门(11题)

文章目录1.删除排序数组中的重复项2.买卖股票的最佳时机 II3.旋转数组4.存在重复元素5.找出只出现一次的元素6.两个数组的交集7.移动零8.加一9.两数之和10.有效的数独11.旋转图像&#x1f48e;&#x1f48e;&#x1f48e;&#x1f48e;&#x1f48e; 更多资源链接&#xff0c;欢…

【手写 Vue2.x 源码】第十八篇 - 根据 render 函数,生成 vnode

一&#xff0c;前言 上篇&#xff0c;介绍了render 函数的生成&#xff0c;主要涉及以下两点&#xff1a; 使用 with 对生成的 code 进行一次包装将包装后的完整 code 字符串&#xff0c;通过 new Function 输出为 render 函数 本篇&#xff0c;根据 render 函数&#xff0c…

linux系统中QT里面的视频播放器的实现方法

大家好&#xff0c;今天主要和大家聊一聊&#xff0c;如何使用QT中视频播放器的方法。 目录 第一&#xff1a;视频播放器基本简介 第二&#xff1a;视频播放器头文件说明 第三&#xff1a;源文件的具体实现方法 第四&#xff1a;运行效果显示 第一&#xff1a;视频播放器基本…

ADS振铃仿真

目录 无振铃时的原理图 无振铃时的Vout和VL输出波形 ​LineCalc对微带线阻抗的计算结果 将微带线线宽Width统一由116改为130 将微带线线宽Width统一由116改为80 将微带线TL9线宽由116改为300 将微带线TL9线宽由116改为50 本文介绍了微带线线宽变化时100MHz信号的反射现象…

2023 年 15 大测试自动化趋势

在过去&#xff0c;软件测试只是为了发现软件产品中的错误。目标是——提高软件质量。但如今&#xff0c;软件测试的范围已经扩大。在软件测试方面&#xff0c;自动化测试一直走在前列。按照最新的测试自动化趋势&#xff0c;软件测试行业有望比过去十年发展得更快。 根据 Mar…

Java面向对象综合训练

Java面向对象综合训练一、文字版格斗游戏Role类测试类输出结果二、对象数组练习对象数组1商品类测试类输出结果对象数组2汽车类测试类输出结果对象数组3手机类测试类输出结果对象数组4女朋友类测试类输出结果对象数组5学生类测试类输出结果一、文字版格斗游戏 Role类 import j…

去掉 域名后面的 /#/ vue-router 和 hbuilder发布 web项目和h5项目

1. vue-router vue-router默认的路由模式是hash&#xff0c;我们要去掉url中的#需要将路由模式切换为history const router new VueRouter({base: test, // 如果项目项目在 域名 根目录下&#xff0c;则去掉这行mode: history, // 路由模式... })这样子&#xff0c;url中的#…

为什么ERP和项目管理的集成是必要的?

在一个企业中&#xff0c;传统的责任分工意味着会计人员看管资金和维持财务标准&#xff0c;而职能经理分配人力资源和维持技术标准。项目经理指导分配的资金和其他资源&#xff0c;同时努力实现项目目标。每个学科都有自己的业务规则&#xff0c;自己的做法&#xff0c;自己的…

C++ | 左值、右值、将亡值和引用的概念 | 聊聊我对它们的深入理解

文章目录前言左右值的辨析一个特殊的问题将亡值引用的深刻理解前言 这篇文章是我在探究完美转发这个语法点时&#xff0c;引发的相关问题思考&#xff0c;为了使自己的理解更深刻&#xff0c;故写下这篇博客 左右值的辨析 首先需要明白两个概念&#xff1a;类型&#xff08;…

1577_AURIX_TC275_MTU中检测控制相关寄存器

全部学习汇总&#xff1a; GreyZhang/g_TC275: happy hacking for TC275! (github.com) 开篇介绍的功能室之前看过很多次的一个握手的功能。快速行以及快速列模式的测试中&#xff0c;这个行列其实是对应的存储的bit阵列信息。一个对应相应的字&#xff0c;另一个则对应bit序列…

【Linux】进程创建、终止、等待、替换、shell派生子进程的理解…

柴犬&#xff1a; 你好啊&#xff0c;屏幕前的大帅哥or大美女&#xff0c;和我一起享受美好的今天叭&#x1f603;&#x1f603;&#x1f603; 文章目录一、进程创建1.调用fork之后&#xff0c;内核都做了什么&#xff1f;2.如何理解fork函数有两个返回值&#xff1f;3.如何理…

(短信服务)java SpringBoot 阿里云短信功能实现发送手机验证码

一.阿里云准备工作 1.阿里云短信服务-注册账号 阿里云官网: https://www.aliyun.com/ 点击官网首页注册按钮。 2.阿里云短信服务-设置短信签名&#xff08;阿里云提供测试的签名&#xff0c;暂时可以跳过&#xff09; 注册成功后&#xff0c;点击登录按钮进行登录。登录后…

简单方式调用WebService服务

好久没有进行过WebService开发了&#xff0c;由于项目需要&#xff0c;重拾WebService&#xff0c;记录一下简单的服务调用方法。拿到需求&#xff0c;仅半页word&#xff0c;其他的就没有了&#xff0c;为了快速开发&#xff0c;尝试过使用插件逆向生成调用的一大堆类&#xf…

AWVS安装与激活

AWVS安装与激活 1.AWVS简介 AWVS&#xff08;Acunetix Web Vulnerability Scanner&#xff09;是一款知名的网络漏洞扫描工具&#xff0c;通过网络爬虫测试网站安全&#xff0c;检测流行的Web应用攻击&#xff0c;如跨站脚本、sql 注入等。据统计&#xff0c;75% 的互联网攻击…

pmp备考全攻略

我这里分享一下我备考的经验&#xff0c;如何对大家有帮助也可以稍微给点支持&#xff0c;让更多人了解&#xff01; 一&#xff0c;我的pmp备考经验 1.一阶段&#xff1a;铺底&#xff0c;花费时间1.5周左右 主要是熟悉考试框架和内容&#xff0c;通过看网盘资料里的章节重…

vue3+ts实现自定义按钮导航

效果图 点击对应按钮&#xff0c;相应按钮被激活&#xff0c;背景平移至激活按钮&#xff0c;字体高亮&#xff0c;其余按钮重置&#xff0c;由于ele没有类似tab&#xff0c;就简单记录下。 实现 <template><div class"tab_wrapper"><spanv-for&q…

这些技巧你值得学会

技巧一&#xff1a;多图合并为PDF文件 处理合并多份PDF文件外&#xff0c;使用PS的【PDF演示文稿】工具&#xff0c;也能一同将多张图片合并成PDF文档&#xff01;通过合并的方式&#xff0c;不但能够批量归纳汇总图片&#xff0c;而且还能根据自身需求&#xff0c;将图片与PD…

Visual studio C++程序内使用Sqlite3

Visual studio C程序内使用Sqlite3 前言 本篇讲解了如何在Visual studio开发的C桌面应用程序内使用Sqlite数据库&#xff0c;Sqlite的语法和Mysql是一样的&#xff0c;所以本篇文章不对数据库语法做过多介绍&#xff0c;介绍一些常用Sqlite的API ★提高阅读体验★ &…

二叉树常见题目

目录 一、判断一棵树是否为另一棵树的子树 二、判断是否对称二叉树 三、翻转二叉树 四、二叉树构建及遍历 五、根据二叉树创建字符串 六、二叉树的最近公共祖先 七、根据前序遍历和中序遍历构造二叉树 八、根据后序遍历和中序遍历构造二叉树 九、二叉树前序非递归…