linux---多线程

  1. 线程的基本概念
    • 定义:在Linux中,线程是进程内部的一个执行单元,是进程的一个实体,它是CPU调度和分派的基本单位。一个进程可以包含多个线程,这些线程共享进程的资源,如代码段、数据段、打开的文件、信号处理函数等,但每个线程都有自己独立的栈空间和程序计数器(PC)。
    • 与进程的区别
      • 进程是资源分配的基本单位,每个进程都有独立的地址空间和系统资源。而线程是进程中的执行路径,共享进程的大部分资源,这使得线程间的通信和切换成本相对较低。
      • 进程间的切换需要进行系统调用,涉及到用户态和内核态的切换,开销较大。线程切换主要是在用户态下进行,只需要保存和恢复少量的寄存器内容和栈指针,速度更快。
  2. 线程库的选择 - pthread库
    • 简介:在Linux系统中,最常用的线程库是POSIX线程库(pthread)。它提供了一系列函数来创建、管理和同步线程。使用pthread库时,需要在编译时链接-lpthread选项。
  3. 线程的创建 - pthread_create函数
    • 函数原型int pthread_create(pthread_t *thread, const pthread_attr_t *attr, void *(*start_routine)(void*), void *arg);
    • 参数说明
      • thread:这是一个指向pthread_t类型变量的指针,用于存储新创建线程的标识符。
      • attr:这是一个指向pthread_attr_t类型的指针,用于设置线程的属性,如线程的栈大小、调度策略等。如果为NULL,则使用默认属性创建线程。
      • start_routine:这是一个函数指针,指向线程的起始函数。线程创建成功后,会从这个函数开始执行。该函数的参数是一个void*类型的指针,返回值也是一个void*类型的指针。
      • arg:这是传递给start_routine函数的参数,通过void*类型的指针传递,可以传递任何类型的数据,在start_routine函数内部需要进行适当的类型转换。
    • 返回值
      • 成功时返回0,表示线程创建成功。
      • 失败时返回一个错误码,并且不会创建线程。可以通过strerror函数将错误码转换为对应的错误信息。
    • 示例代码(创建一个简单的线程)
#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>
// 线程执行的函数
void *thread_function(void *arg) {// 在这里编写线程要执行的任务printf("这是一个新线程。\n");// 线程结束时返回NULLreturn NULL;
}
int main() {pthread_t thread_id;int result;// 创建线程result = pthread_create(&thread_id, NULL, thread_function, NULL);if (result!= 0) {// 如果创建线程失败,打印错误信息perror("线程创建失败");return 1;}// 主线程继续执行其他任务,这里简单地等待新线程结束result = pthread_join(thread_id, NULL);if (result!= 0) {perror("等待线程结束失败");return 1;}printf("主线程继续执行,线程已结束。\n");return 0;
}
  1. 线程的终止 - pthread_exit函数和其他方式
    • pthread_exit函数

      • 函数原型void pthread_exit(void *value_ptr);
      • 功能:用于显式地终止一个线程。当线程执行到pthread_exit函数时,线程会立即终止,并将value_ptr指向的值返回给等待该线程结束的线程(如果有)。这个返回值可以通过pthread_join函数获取。
        在这里插入图片描述
    • start_routine函数中return返回

      • 线程执行的起始函数start_routine执行return语句时,线程也会正常终止,返回值的处理方式和pthread_exit类似。
    • 线程被其他线程取消(pthread_cancel函数)

      • 函数原型int pthread_cancel(pthread_t thread);
      • 功能:一个线程可以通过pthread_cancel函数来请求取消另一个线程。被取消的线程会在合适的时机(如在某些系统调用中或者在检查取消点时)终止。不过,线程可以通过设置一些属性来控制是否响应取消请求以及如何响应。
  2. 线程的等待 - pthread_join函数
    • 函数原型int pthread_join(pthread_t thread, void **value_ptr);
    • 参数说明
      • thread:要等待的线程的标识符。
      • value_ptr:这是一个指向void*类型指针的指针,用于存储被等待线程的返回值(如果线程通过pthread_exit函数或者在start_routine函数中return返回了一个值)。如果不需要获取返回值,可以将这个参数设置为NULL
    • 返回值
      • 成功时返回0,表示成功等待线程结束并获取了返回值(如果需要获取)。
      • 失败时返回一个错误码,并且不会正确等待线程结束。
  3. 线程的同步 - 互斥锁(Mutex)
    • 基本概念
      • 互斥锁用于保护共享资源,防止多个线程同时访问和修改共享资源而导致数据不一致。互斥锁有两种状态:锁定和解锁。当一个线程获取了互斥锁(锁定状态)后,其他线程如果试图获取同一把互斥锁,就会被阻塞,直到该锁被释放(解锁状态)。
    • 互斥锁相关函数
      • pthread_mutex_init函数:用于初始化一个互斥锁。函数原型为int pthread_mutex_init(pthread_mutex_t *mutex, const pthread_mutexattr_t *attr);,其中mutex是指向pthread_mutex_t类型变量的指针,用于存储互斥锁,attr是指向互斥锁属性的指针,如果为NULL,则使用默认属性初始化。
      • pthread_mutex_lock函数:用于获取(锁定)互斥锁。函数原型为int pthread_mutex_lock(pthread_mutex_t *mutex);,如果互斥锁已经被其他线程锁定,调用该函数的线程会被阻塞,直到互斥锁被释放。
      • pthread_mutex_unlock函数:用于释放(解锁)互斥锁。函数原型为int pthread_mutex_unlock(pthread_mutex_t *mutex);,当一个线程完成对共享资源的访问后,应该及时释放互斥锁,以便其他线程可以获取。
      • pthread_mutex_destroy函数:用于销毁互斥锁。函数原型为int pthread_mutex_destroy(pthread_mutex_t *mutex);,在互斥锁不再使用时,应该销毁它,释放相关资源。
    • 示例代码(使用互斥锁保护共享资源)
#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>
// 定义一个共享变量
int shared_variable = 0;
// 定义互斥锁
pthread_mutex_t mutex;
// 线程执行的函数
void *thread_function(void *arg) {int i;for (i = 0; i < 1000; ++i) {// 锁定互斥锁pthread_mutex_lock(&mutex);// 访问共享资源shared_variable++;// 解锁互斥锁pthread_mutex_unlock(&mutex);}return NULL;
}
int main() {pthread_t thread_id1, thread_id2;int result;// 初始化互斥锁result = pthread_mutex_init(&mutex, NULL);if (result!= 0) {perror("互斥锁初始化失败");return 1;}// 创建两个线程result = pthread_create(&thread_id1, NULL, thread_function, NULL);if (result!= 0) {perror("线程1创建失败");return 1;}result = pthread_create(&thread_id2, NULL, thread_function, NULL);if (result!= 0) {perror("线程2创建失败");return 1;}// 等待线程结束result = pthread_join(thread_id1, NULL);if (result!= 0) {perror("等待线程1结束失败");return 1;}result = pthread_join(thread_id2, NULL);if (result!= 0) {perror("等待线程2结束失败");return 1;}// 销毁互斥锁result = pthread_mutex_destroy(&mutex);if (result!= 0) {perror("互斥锁销毁失败");return 1;}printf("共享变量的值为: %d\n", shared_variable);return 0;
}
  1. 线程的同步 - 条件变量(Condition Variable)
    • 基本概念
      • 条件变量用于在线程之间进行同步,它允许一个线程等待某个条件为真。通常与互斥锁一起使用,一个线程可以在满足某个条件时通知其他等待该条件的线程。
    • 条件变量相关函数
      • pthread_cond_init函数:用于初始化一个条件变量。函数原型为int pthread_cond_init(pthread_cond_t *cond, const pthread_condattr_t *attr);,其中cond是指向pthread_cond_t类型变量的指针,用于存储条件变量,attr是指向条件变量属性的指针,如果为NULL,则使用默认属性初始化。
      • pthread_cond_wait函数:用于让一个线程等待条件变量满足。函数原型为int pthread_cond_wait(pthread_cond_t *cond, pthread_mutex_t *mutex);,在调用该函数时,线程会释放已经获取的互斥锁mutex,然后进入等待状态,直到另一个线程通过pthread_cond_signal或者pthread_cond_broadcast函数唤醒它,并且在被唤醒后会重新获取互斥锁。
      • pthread_cond_signal函数:用于唤醒一个等待条件变量的线程。函数原型为int pthread_cond_signal(pthread_cond_t *cond);,它会唤醒至少一个等待条件变量cond的线程。
      • pthread_cond_broadcast函数:用于唤醒所有等待条件变量的线程。函数原型为int pthread_cond_broadcast(pthread_cond_t *cond);
      • pthread_cond_destroy函数:用于销毁条件变量。函数原型为int pthread_cond_destroy(pthread_cond_t *cond);,在条件变量不再使用时,应该销毁它,释放相关资源。
    • 示例代码(使用条件变量和互斥锁实现线程同步)
#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>
// 定义互斥锁和条件变量
pthread_mutex_t mutex;
pthread_cond_t cond;
// 共享变量,表示资源是否可用
int resource_available = 0;
// 线程1执行的函数,用于生产资源
void *thread1_function(void *arg) {// 模拟生产资源的过程// 这里简单地等待一段时间后生产资源sleep(2);// 锁定互斥锁pthread_mutex_lock(&mutex);resource_available = 1;// 发送信号,通知等待资源的线程pthread_cond_signal(&cond);// 解锁互斥锁pthread_mutex_unlock(&mutex);return NULL;
}
// 线程2执行的函数,用于消费资源
void *thread2_function(void *arg) {// 锁定互斥锁pthread_mutex_lock(&mutex);// 检查资源是否可用,如果不可用则等待while (!resource_available) {pthread_cond_wait(&cond, &mutex);}// 消费资源,这里简单地打印一条消息printf("资源已被消费。\n");// 解锁互斥锁pthread_mutex_unlock(&mutex);return NULL;
}
int main() {pthread_t thread_id1, thread_id2;int result;// 初始化互斥锁和条件变量result = pthread_mutex_init(&mutex, NULL);if (result!= 0) {perror("互斥锁初始化失败");return 1;}result = pthread_cond_init(&cond, NULL);if (result!= 0) {perror("条件变量初始化失败");return 1;}// 创建两个线程result = pthread_create(&thread_id1, NULL, thread1_function, NULL);if (result!= 0) {perror("线程1创建失败");return 1;}result = pthread_create(&thread_id2, NULL, thread2_function, NULL);if (result!= 0) {perror("线程2创建失败");return 1;}// 等待线程结束result = pthread_join(thread_id1, NULL);if (result!= 0) {perror("等待线程1结束失败");return 1;}result = pthread_join(thread_id2, NULL);if (result!= 0) {perror("等待线程2结束失败");return 1;}// 销毁互斥锁和条件变量result = pthread_mutex_destroy(&mutex);if (result!= 0) {perror("互斥锁销毁失败");return 1;}result = pthread_cond_destroy(&cond);if (result!= 0) {perror("条件变量销毁失败");return 1;}return 0;
}

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

将4G太阳能无线监控的视频接入电子监控大屏,要考虑哪些方面?

随着科技的飞速发展&#xff0c;4G太阳能无线监控系统以其独特的优势在远程监控领域脱颖而出。这种系统结合了太阳能供电的环保特性和4G无线传输的便捷性&#xff0c;为各种环境尤其是无电或电网不稳定的地区提供了一种高效、可靠的视频监控解决方案。将这些视频流接入大屏显示…

有监督学习 vs 无监督学习:机器学习的两大支柱

有监督学习 vs 无监督学习&#xff1a;机器学习的两大支柱 有监督学习 vs 无监督学习&#xff1a;机器学习的两大支柱一、有无“老师”来指导二、解决的问题类型不同三、模型的输出不同 有监督学习 vs 无监督学习&#xff1a;机器学习的两大支柱 在机器学习的奇妙世界里&#…

SLURM资料

SLURM资料 Quick Start 基本概念 job step&#xff1a; 作业步&#xff0c;单个作业可以有多个作业步partition&#xff1a;分区&#xff0c;作业需要在特定分区中运行&#xff08;理解为定义了队列&#xff0c;每个队列中包含不同节点&#xff09;QOS&#xff1a;服务质量&a…

App自动化之dom结构和元素定位方式(包含滑动列表定位)

DOM结构 先来看几个名词和解释&#xff1a; dom: Document Object Model 文档对象模型 dom应用: 最早应用于html和js的交互。界面的结构化描述&#xff0c; 常见的格式为html、xml。核心元素为节点和属性 xpath: xml路径语言&#xff0c;用于xml 中的节点定位&#xff0c;X…

Vulhub:Redis[漏洞复现]

4-unacc(Redis未授权代码执行) 启动漏洞环境 docker-compose up -d 阅读vulhub给出的漏洞文档 cat README.zh-cn.md # Redis 4.x/5.x 主从复制导致的命令执行 Redis是著名的开源Key-Value数据库&#xff0c;其具备在沙箱中执行Lua脚本的能力。 Redis未授权访问在4.x/5.0.5以…

imx6ull qt多页面控制系统(正点原子imx系列驱动开发)

开题答辩完了也考完了四六级&#xff0c;赶紧来更新一下一个月前留下的坑吧 QAQ首先&#xff0c;因为毕业设计需要用到这些知识所以就从网络上找了一个智能车机系统&#xff0c;借鉴了一下大佬的项目思路&#xff0c;缝缝补补一个月终于完成了这一内容。 在这里先感谢从两位大佬…

前端小白学习之路-Vben探索 vite 配置 - 1/50

目的 为ApiHug 寻找一个前端解决方案前端背景知识缺乏整盘操作&#xff1a;前后全栈80% 中小规模项目提效 30% 全员全栈快速构建高度模块化AI Native... 所以 裸学前端高举高打&#xff0c;直接从复杂项目拆解AI 助手高度依赖后端癖严重&#xff0c;高度模块&#xff0c; 结构化…

Docker:Dockerfile(补充四)

这里写目录标题 1. Dockerfile常见指令1.1 DockerFile例子 2. 一些其他命令 1. Dockerfile常见指令 简单的dockerFile文件 FROM openjdk:17LABEL authorleifengyangCOPY app.jar /app.jarEXPOSE 8080ENTRYPOINT ["java","-jar","/app.jar"]# 使…

谷歌浏览器的扩展市场使用指南

谷歌浏览器的扩展市场为用户提供了丰富多样的功能扩展&#xff0c;可以大幅提升浏览体验。本文将为你详细介绍如何使用谷歌浏览器的扩展市场&#xff0c;包括安装、管理和一些推荐的无障碍工具、图标重置方法和便捷操作技巧。&#xff08;本文由https://chrome.py010.cn/的作者…

04、Vue与Ajax

4.1 发送AJAX异步请求的方式 发送AJAX异步请求的常见方式包括&#xff1a; 4.1.1. 原生方式 使用浏览器内置的JS对象XMLHttpRequest const xhr new XMLHttpRequest() xhr.open() xhr.send() xhr.onreadystatechange function(){} 4.1.2. 原生方式 使用浏览器内置的JS函…

网络安全概论——防火墙原理与设计

一、防火墙概述 防火墙是一种装置&#xff0c;它是由软件/硬件设备组合而成&#xff0c;通常处于企业的内部局域网与 Internet 之间&#xff0c;限制 Internet 用户对内部网络的访问以及管理内部用户访问 Internet 的权限。换言之&#xff0c;一个防火墙在一个被认为是安全和可…

南城云趣:智能云平台,杜绝电动车充电安全隐患

电动自行车作为绿色低碳出行的主要方式之一,受到无数市民的推崇,而电动自行车数量的急剧上涨,也严重增加小区管理的负担。记者调查发现,目前电动自行车缺乏有效的管理,使得带车或电瓶上楼充电、乱停乱放、车辆容易被盗等安全问题日益突出,给社区消防安全和管理带来严峻的挑战。…

Linux 文件系统目录结构及其简要介绍

&#x1f44b; 欢迎来到“Linux学习&#xff1a;Linux 文件系统目录结构”篇&#xff01; 接下来让我们一起来学习一下Linux 文件系统目录结构吧&#xff01;祝你有所收获&#xff01; 文章目录 总结表格Linux 文件系统目录结构及其简要介绍补充小资源 小伙伴们都知道&#xff…

【服务器】MyBatis是如何在java中使用并进行分页的?

MyBatis 是一个支持普通 SQL 查询、存储过程和高级映射的持久层框架。它消除了几乎所有的 JDBC 代码和参数的手动设置以及结果集的检索。MyBatis 可以通过简单的 XML 或注解来配置和映射原始类型、接口和 Java 的 POJO&#xff08;Plain Old Java Objects&#xff0c;普通老式 …

Elasticsearch-DSL高级查询操作

一、禁用元数据和过滤数据 1、禁用元数据_source GET product/_search {"_source": false, "query": {"match_all": {}} }查询结果不显示元数据 禁用之前: {"took" : 0,"timed_out" : false,"_shards" : {&quo…

使用 UniApp 在微信小程序中实现 SSE 流式响应

概述 服务端发送事件(Server-Sent Events, SSE)是一种允许服务器向客户端推送实时更新的技术。SSE 提供了一种单向的通信通道,服务器可以持续地向客户端发送数据,而不需要客户端频繁发起请求。这对于需要实时更新的应用场景非常有用。 流式传输的特点是将数据逐步传输给客…

【Tomcat】第六站(最后一站啦!):数据的返回

1. 引言 前端资源比如html页面&#xff0c;进行返回。截止到目前我们写的项目架构不支持前端页面&#xff08;静态资源 &#xff09;。 2. 数据的返回 2.1 准备 为了能够写前端页面&#xff0c;新建一个项目。选择Maven项目&#xff0c;下一步&#xff0c;下一步。 加载完…

electron-vite【实战系列教程】

创建项目 https://blog.csdn.net/weixin_41192489/article/details/144442262 安装必要的插件 UI 库 element-plus npm install element-plus --save安装 element-plus 图标 npm install element-plus/icons-vue安装插件 – 自动注册组件 vs 自动导入框架方法 npm install -…

信号处理相关的东东(学习解惑)

信号处理相关的东东&#xff08;学习解惑&#xff09; 所有内容学习自知乎专栏&#xff0c;https://www.zhihu.com/column/xinhao&#xff0c;写的很好&#xff0c;值得反复学习 时频域分析的一些常用概念 FROM&#xff1a;https://zhuanlan.zhihu.com/p/35742606 1、相加性…

[Python学习日记-73] 面向对象实战1——答题系统

[Python学习日记-73] 面向对象实战1——答题系统 简介 需求模型——5w1h8c 领域模型 设计模型 实现模型 案例&#xff1a;年会答题系统 简介 在学习完面向对象之后你会发现&#xff0c;你还是不会自己做软件做系统&#xff0c;这是非常正常的&#xff0c;这是因为计算机软…