Linux系统编程 多线程基础

news2025/1/17 21:54:18

文章目录

  • 前言
  • 一、线程概念
  • 二、线程的创建
  • 三、线程的退出
  • 四、pthread_join函数
  • 总结


前言

本篇文章作为多线程的入门讲解将带大家先创建几个线程来感受一下什么是多线程,了解一下多线程到底有什么作用。

一、线程概念

线程(Thread)是程序执行流的最小单元,是进程中的一个实体,是被操作系统独立调度和分派 CPU 时间的基本单位。线程和进程一样都属于操作系统中的多任务处理机制。

每一个线程都有自己独立的运行栈和程序计数器,并且共享所属进程的内存空间。线程共享的资源包括代码段、数据段和打开的文件等。线程的创建和撤销的开销比进程要小,因此多线程并发通常比多进程更加高效。

线程可以完成各种任务,如运算、时间操作、等待、响应中端请求等。它可以在同一时间内完成多项活动,实现真正的并发操作。线程通常由一个线程函数和一些传递给该函数的参数组成,线程函数是线程的执行体,它会不断的执行,直到结束或等待某些事件或资源。线程可以通过互斥锁等同步机制来访问共享资源,从而保证线程执行的正确性和可靠性。

一个进程在某一时刻只能做一件事情,有了多线程后,就可以把进程设计成在某一时刻能够不止做一件事,每个线程处理各自的独立的任务。

二、线程的创建

在man手册中我们可以查看到创建线程的函数:
在这里插入图片描述
创建线程函数原型:

int pthread_create(pthread_t *thread, const pthread_attr_t *attr,
                          void *(*start_routine) (void *), void *arg);

各个参数解析:

thread:指向pthread_t类型的指针,用于存储新线程的ID。

attr:指向pthread_attr_t类型的指针,用于设置新线程的属性。可以通过该参数指定线程的分离状态、栈大小等属性。如果不需要指定属性,可以传递NULL。

start_routine:指向线程执行体的函数指针。新线程会从该函数的起始地址开始执行。

arg:传递给线程执行体的参数,可以是任意类型的指针。该参数会作为start_routine的参数,可以在start_routine函数中进行处理。

创建一个线程代码:

#include <pthread.h>
#include <stdio.h>
#include <unistd.h>

void* Thread1(void *arg)
{
    while (1)
    {
        printf("this is Thread1\n");

        sleep(1);//休眠1s
    }
    
}


int main(void)
{
    pthread_t tid;//线程ID
    int err = -1;

    err = pthread_create(&tid, NULL, Thread1, NULL);

    if(err < 0)
    {
        printf("create thread is err\n");
    }


    while (1)
    {
        printf("this is Main Thread\n");

        sleep(1);//休眠1s
    }
    


    return 0;
}

运行结果:
主线程和被创建的线程轮流运行,互相不会干扰。
在这里插入图片描述

三、线程的退出

线程退出函数原型:

void pthread_exit(void *value_ptr);

value_ptr:表示线程的返回值指针,用于存储线程的返回值。如果不需要返回值,可以传递NULL。

在一个线程中,调用pthread_exit函数可以立即终止该线程的执行,并将value_ptr指向的值作为线程的返回值。该返回值可以被其他线程使用pthread_join函数获取。如果当前线程是主线程,则该函数会终止整个程序的执行。

需要注意的是,线程在终止时一定要记得调用pthread_exit函数来释放它所持有的资源,以免导致资源泄露。如果线程没有调用pthread_exit而是直接返回,那么线程持有的资源将无法被正确释放,造成内存泄漏等问题。

示例代码:

#include <pthread.h>
#include <stdio.h>
#include <unistd.h>

void* Thread1(void *arg)
{
    static int i = 0;
    while (1)
    {
        printf("this is Thread1\n");

        if(++i == 5)
        {
            pthread_exit(NULL);
        }

        sleep(1);//休眠1s
    }
    
}


int main(void)
{
    pthread_t tid;//线程ID
    int err = -1;

    err = pthread_create(&tid, NULL, Thread1, NULL);

    if(err < 0)
    {
        printf("create thread is err\n");
    }


    while (1)
    {
        printf("this is Main Thread\n");

        sleep(1);//休眠1s
    }
    


    return 0;
}

运行效果:
通过效果我们可以知道使用pthread_exit能够退出线程并释放线程占用的资源。
在这里插入图片描述

四、pthread_join函数

函数原型:

int pthread_join(pthread_t thread, void **value_ptr);

thread:需要等待的线程的ID。

value_ptr:用于存储线程的返回值。该参数是一个指向指针的指针,用于返回线程的返回值。如果不需要返回值,可以传递NULL。

函数返回值为0表示成功等待线程的结束并获取返回值,负数则表示等待线程失败。

在一个线程中,调用pthread_join可以等待指定的线程结束并获取它的返回值。如果调用pthread_join的线程已经结束,而被等待的线程还在运行,则会立即返回并不会等待。如果被等待的线程已经运行结束,则可以通过value_ptr参数获取该线程的返回值。如果不关注该线程的返回值,则可以将value_ptr参数设置为NULL。

需要注意的是,如果被等待的线程没有通过pthread_exit来结束执行,而是使用return语句,那么在等待结束时,其返回值可能无法被正确获取,甚至可能出现未定义行为。此外,如果线程已经被分离,那么调用pthread_join将会失败。

示例代码:

使用pthread_join等待Thread1这个线程执行结束后再退出主线程。

#include <pthread.h>
#include <stdio.h>
#include <unistd.h>

void* Thread1(void *arg)
{
    static int i = 0;
    while (1)
    {
        printf("this is Thread1\n");

        if(++i == 5)
        {
            pthread_exit(NULL);
        }

        sleep(1);//休眠1s
    }
    
}


int main(void)
{
    pthread_t tid;//线程ID
    int err = -1;

    err = pthread_create(&tid, NULL, Thread1, NULL);

    if(err < 0)
    {
        printf("create thread is err\n");
    }

    pthread_join(tid, NULL);

    
    printf("Main thread exit\n");

    return 0;
}

在这里插入图片描述

总结

本篇文章介绍了线程的创建和退出等简单操作,带大家开启了Linux多线程的学习,大家有什么疑问可以跟我留言或者评论。

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

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

相关文章

C++类与对象(二)——构造函数与析构函数

文章目录 一.类的默认6个成员函数二.构造函数1.引例2.构造函数的概念及特性 三.析构函数&#x1f60b;析构函数的特性 前言&#xff1a; 上篇文章初步认识了类以及类的相关知识&#xff0c;本篇将继续深入学习类与对象——类的默认6个成员函数&#xff1a; 一.类的默认6个成员函…

制造企业如何跨越大规模定制鸿沟?中国最大减速机企业的答案来了

导读&#xff1a;传统制造企业如何深度用云&#xff1f; 在中国制造向中国智造的转型中&#xff0c;长三角地区一直扮演着急先锋的角色。总部位于常州的江苏国茂减速机股份有限公司(简称国茂股份)&#xff0c;就是中国制造上云转型的典型代表。 国茂股份成立于1993年&#xff0…

Microelectronic学习章节总结(2)-- data path和control unit设计

文章目录 part1. Data path设计1.1 logic unit1.2 shifter1.3 adder1.4 comparator1.5 multiplier1.6 divider1.7 register file part2. Control unit设计part3. CPU SoC上的其它部件 (TODO&#xff1a;理解总结)3.1 总线AMBA&#xff08;parallel&#xff09; 3.2 Memorymemor…

[C++基础]-类和对象(下)

前言 作者&#xff1a;小蜗牛向前冲 名言&#xff1a;我可以接受失败&#xff0c;但我不能接受放弃 如果觉的博主的文章还不错的话&#xff0c;还请点赞&#xff0c;收藏&#xff0c;关注&#x1f440;支持博主。如果发现有问题的地方欢迎❀大家在评论区指正。 目录 一、深入学…

推荐算法实战项目:AFM 原理以及案例实战(附完整 Python 代码)

本文要介绍的是由浙江大学联合新加坡国立大学提出的AFM模型。通过名字也可以看出&#xff0c;此模型又是基于FM模型的改进&#xff0c;其中A代表”Attention“&#xff0c;即AFM模型实际上是在FM模型中引入了注意力机制改进得来的。 之所以要在FM模型中引入注意力机制&#xf…

PC3-管理员操作

token无效可能&#xff0c;就是token过期了需要配置&#xff1a;&#xff1a;&#xff1a; history 安装可以跳路由在ts文件中&#xff1a;因为在ts文件中还需要store&#xff0c;清空token // 安装可以跳路由在ts文件中npm i history 防止接口出现 token 无效&#xff0c;登…

【C++】AVL平衡二叉树源码剖析

目录 概述 算法 左单旋 右单旋 左右双旋 右左双旋 源码 AVLTree.h test.cpp 概述 AVL树也叫平衡二叉搜索树&#xff0c;是二叉搜索树的进化版&#xff0c;设计是原理是弥补二叉搜索树的缺陷&#xff1a;当插入的数据接近于有序数列时&#xff0c;二叉搜索树的性能严重…

20天能拿下PMP吗?

新版大纲&#xff0c;专注于人员、过程、业务环境三个领域&#xff0c;内容贯穿价值交付范围&#xff08;包括预测、敏捷和混合的方法&#xff09;。除了考试时间由240分钟变更为230分钟、200道单选题变为180道&#xff08;包含单选和多选&#xff09;之外&#xff0c;新考纲还…

【Ubuntu18配置Anaconda深度学习环境】

参考&#xff1a;Ubuntu18配置与ROS 兼容的深度学习环境&#xff08;Anaconda3PyTorch1.10python3.8cuda10.2&#xff09; 1. 前言 之前在Window下安装了Anaconda&#xff0c;熟悉了一下安装过程&#xff0c;Ubuntu18.04下最难的应该就是和ROS Melodic的兼容问题。ROS1是基于P…

Linux常用命令——inotifywait命令

在线Linux命令查询工具 inotifywait 异步文件系统监控机制 补充说明 Inotify一种强大的、细粒度的、异步文件系统监控机制&#xff0c;它满足各种各样的文件监控需要&#xff0c;可以监控文件系统的访问属性、读写属性、权限属性、删除创建、移动等操作&#xff0c;也就是可…

数据结构之第十一章、排序算法

一、排序的概念及引用 1.1排序的概念 排序&#xff1a;所谓排序&#xff0c;就是使一串记录&#xff0c;按照其中的某个或某些关键字的大小&#xff0c;递增或递减的排列起来的操作。 1.1.1排序的稳定性 稳定性&#xff1a;假定在待排序的记录序列中&#xff0c;存在多个具…

数据结构之第九章、优先级队列(堆)

目录 一、优先级队列 1.1概念 二、优先级队列的模拟实现 2.1堆的概念 2.2堆的存储方式 2.3堆的创建 2.3.1堆向下调整 2.3.2堆的创建 2.3.3建堆的时间复杂度 2.4堆的插入与删除 2.4.1堆的插入 2.4.2堆的删除 2.5用堆模拟实现优先级队列 三、常用接口介绍 3.1Priori…

计算机组成原理与体系结构

目录 第一章、计算机组成原理与体系结构1、数据的表示1.1.进制转换1.2.原码、反码、补码、移码1.3.数据的表述 2、计算机结构3、Flynn分类法4、CISC与RISC5、流水线技术5.1、流水线的基本概念5.2、流水线的计算5.3、流水线吞吐率计算5.4、流水线加速比计算5.5、流水线的效率 6、…

Python小姿势 - Python的多线程编程

Python的多线程编程 Python的多线程编程提供了一个非常简单的方法来让一个Python程序同时运行多个任务。这个方法通过创建新的线程来实现&#xff0c;线程可以被视为一个单独的执行流程。 为了创建一个新线程&#xff0c;我们需要使用Python的_thread模块中的start_new_thread(…

【IDEA】简单入门:请求数据库表数据

目录 修改编辑与控制台字体大小 二、sprintboot项目入门 【1】直接开始配置Controller 【2】直接请求数据库中的数据&#xff0c;返回json格式 &#xff08;0&#xff09;整合PostgreSQL框架 &#xff08;2&#xff09;实体entity类 &#xff08;3&#xff09;控制类Mai…

快速了解车联网V2X通信

自动驾驶拥有极其巨大的潜力&#xff0c;有可能改变我们的出行方式。它不仅有望永远改变车辆的设计和制造&#xff0c;还会永远改变汽车的所有权乃至整个交通运输业务。要实现全自动驾驶的目标&#xff0c;开发人员需要开发极为复杂的软件&#xff0c;软件中融入的人工智能(AI)…

机械硬盘和固态硬盘有什么区别?如何使用?

案例&#xff1a;怎么区分机械硬盘和固态硬盘&#xff1f; 【我知道硬盘可以用来储存数据&#xff0c;但我不知道机械硬盘和固态硬盘的区别&#xff0c;有没有小伙伴可以详细解释一下。】 硬盘可以用来储存数据&#xff0c;常见的硬盘有两种&#xff0c;分别是机器硬盘和固态…

C++11多线程编程——线程池的实现

学一门新技术&#xff0c;还是要问那个问题&#xff0c;为什么我们需要这个技术&#xff0c;这个技术能解决什么痛点。 一、为何需要线程池 那么为什么我们需要线程池技术呢&#xff1f;多线程编程用的好好的&#xff0c;干嘛还要引入线程池这个东西呢&#xff1f;引入一个新的…

发展文旅夜游项目有哪些好处

夜晚的城市&#xff0c;总是充满着无限的魅力和活力&#xff0c;而文旅夜游更是让这份魅力和活力得到了更好的展现和发挥。新起典文旅科技认为文旅夜游不仅仅是一种旅游方式&#xff0c;更是可以增加城市夜间经济、丰富文化娱乐生活、缓解白天拥堵、提高旅游体验、促进文化交流…

HTTP的特点

灵活可扩展 HTTP 协议最初诞生的时候就比较简单&#xff0c;本着开放的精神只规定了报文的基本格式&#xff0c;比如用空格分隔单词&#xff0c;用换行分隔字段&#xff0c;“headerbody”等&#xff0c;报文里的各个组成部分都没有做严格的语法语义限制&#xff0c;可以由开发…