《Linux C编程实战》笔记:线程同步
这一节主要是解决共享资源的处理。操作系统里也讲过互斥、锁之类的概念。
互斥锁
互斥锁通过锁机制来实现线程同步,同一时刻只允许一个线程执行一个关键部分的代码
一下是操作互斥锁的函数,均声明在pthread.h中。
pthread_mutex_init(初始化互斥锁):
int pthread_mutex_init(pthread_mutex_t *mutex, const pthread_mutexattr_t *attr);该函数用于初始化互斥锁,其中
mutex是指向互斥锁的指针,attr是指向互斥锁属性的指针。如果不需要特殊属性,可以将attr参数设置为NULL。
pthread_mutex_destroy(销毁互斥锁):
int pthread_mutex_destroy(pthread_mutex_t *mutex);该函数用于销毁互斥锁,释放相关资源。在互斥锁使用完毕后,应该调用该函数来进行清理操作。函数成功执行时返回0.
pthread_mutex_lock(加锁):
int pthread_mutex_lock(pthread_mutex_t *mutex);该函数用于尝试获得互斥锁的所有权。如果互斥锁已经被其他线程占用,调用线程将被阻塞,直到互斥锁可用。
pthread_mutex_unlock(解锁):
int pthread_mutex_unlock(pthread_mutex_t *mutex);该函数用于释放互斥锁(与加锁的必须是同一线程),允许其他线程获得该互斥锁的所有权。应该在对共享资源的访问完成后调用该函数。
pthread_mutex_trylock(尝试加锁):
int pthread_mutex_trylock(pthread_mutex_t *mutex);该函数尝试获得互斥锁的所有权,但如果锁已经被其他线程占用,则不会阻塞,而是立即返回错误EBUSY。可以利用这个函数来实现非阻塞的加锁操作。
互斥锁的初始化可以用静态赋值法,将一个宏结构赋给mutex
pthread_mutex_t mutex=PTHREAD_MUTEX_INITIALIZER;
或者是用初始化函数,不过用之前还要先了解pthread_mutexattr_t怎么用。
pthread_mutexattr_t mutexattr;pthread_mutexattr_init(&mutexattr);pthread_mutexattr_settype(&mutexattr,PTHREAD_MUTEX_RECURSIVE);
这样算是设定好了锁的属性。属性一般是以下几种
PTHREAD_MUTEX_NORMAL:
- 普通互斥锁,不允许递归锁。对同一线程多次调用
pthread_mutex_lock会导致死锁。PTHREAD_MUTEX_RECURSIVE:
- 允许同一线程多次获得互斥锁,使用嵌套锁。线程每成功调用一次
pthread_mutex_lock,必须调用相同次数的pthread_mutex_unlock才能释放锁。- 如果是不同线程请求,则在解锁时重新竞争
PTHREAD_MUTEX_ERRORCHECK:
- 提供错误检查,如果同一线程重复调用
pthread_mutex_lock,会返回错误EDEADLK。PTHREAD_MUTEX_DEFAULT:
- 默认类型,通常等同于
PTHREAD_MUTEX_NORMAL。
注意:加锁时,不论哪种类型的锁,都不可能被两个不同的线程同时得到,其中一个必须等待解锁。在同一进程中的线程,如果加锁后没有解锁,则其他线程将无法再获得该锁。
示例代码
以下代码演示了互斥锁保护全局变量的用法
pthread_mutex_t number_mutex;
int globalnumber;
void write_globalnumber(){pthread_mutex_lock(&number_mutex);globalnumber++;pthread_mutex_unlock(&number_mutex);
}
int read_globalnumber(){int temp;pthread_mutex_lock(&number_mutex);temp=globalnumber;pthread_mutex_unlock(&number_mutex);return temp;
}
还是很好懂的,在读写之前上锁,操作完之后解锁。
条件变量
条件变量是利用线程间共享的全局变量进行同步的一种机制。 条件变量宏观上类似if语句,
符合条件就能执行某段程序,否则只能等待条件成立。
使用条件变量主要包括两个动作:一个等待使用资源的线程等待“条件变量被设置为真”;另一个线程在使用完资源后“设置条件为真”,这样就可以保证线程间的同步了。这样就存在一个关键问题,就是要保证条件变量能被正确的修改,条件变量要收到特殊的保护。实际使用中互斥锁扮演者这样的一个保护者的角色。Linux提供了一系列对条件变量操作的函数
pthread_cond_init:
int pthread_cond_init(pthread_cond_t *cond, const pthread_condattr_t *attr);该函数用于初始化条件变量。
pthread_cond_t是条件变量的类型,attr是条件变量的属性,通常可以设置为NULL表示默认属性。pthread_cond_destroy:
int pthread_cond_destroy(pthread_cond_t *cond);该函数用于销毁条件变量。在条件变量不再使用时调用,以释放相关资源。
pthread_cond_wait:
int pthread_cond_wait(pthread_cond_t *cond, pthread_mutex_t *mutex);这个函数使线程等待条件变量的信号。在调用这个函数之前,线程通常需要先获取一个互斥锁(
mutex),然后在等待条件变量的时候会释放这个互斥锁。pthread_cond_timedwait:
int pthread_cond_timedwait(pthread_cond_t *cond, pthread_mutex_t *mutex, const struct timespec *abstime);类似于
pthread_cond_wait,但是它在指定的时间(abstime)之后超时返回。abstime是一个绝对时间。pthread_cond_signal:
int pthread_cond_signal(pthread_cond_t *cond);该函数用于发送信号给一个正在等待条件变量的线程,唤醒其中一个。通常在某个条件变为真的时候调用。
pthread_cond_broadcast:
int pthread_cond_broadcast(pthread_cond_t *cond);该函数用于发送信号给所有正在等待条件变量的线程,唤醒它们。通常在某个条件变为真的时候调用。
和互斥锁一样,条件变量的初始化也可以用静态赋值法
pthread_cond_t cond=PTHREAD_COND_INITIALIZER;
还有一种方式自然是用初始化函数,一般attr都是空指针
pthread_cond_wait函数书房由mutex指向的互斥锁,同时使当前线程关于cond指向的条件变量阻塞,直到条件被信号唤醒。通常条件表达式在互斥锁的保护下求值,如果条件表达式为假,那么线程基于条件变量阻塞。当一个线程改变条件变量的值时,条件变量获得一个信号,使得等待条件变量的线程退出阻塞状态。
线程被条件变量阻塞后,可通过两个函数激活。
signal只激活一个等待的线程,存在多个的话按入队顺序激活;broadcast激活所有等待的线程
清除函数destory只有在没有线程等待该条件变量的时候才能清除,否则返回EBUSY
说了这么多,其实也不知道应该怎么用,看下面的代码
示例代码
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<pthread.h>
pthread_mutex_t mutex=PTHREAD_MUTEX_INITIALIZER;
pthread_cond_t cond=PTHREAD_COND_INITIALIZER;
void cleanup_handler(void *arg) {pthread_mutex_unlock((pthread_mutex_t *)arg);
}
void *thread1(void *arg){pthread_cleanup_push(cleanup_handler,&mutex);//这个在上一篇文章有讲过while (1){printf("thread1 is running\n");pthread_mutex_lock(&mutex);pthread_cond_wait(&cond,&mutex);printf("thread1 applied the condition\n");pthread_mutex_unlock(&mutex);sleep(4);}pthread_cleanup_pop(0);}
void *thread2(void *arg){while(1){printf("thread2 is running\n");pthread_mutex_lock(&mutex);pthread_cond_wait(&cond,&mutex);printf("thread2 applied the condition\n");pthread_mutex_unlock(&mutex);sleep(1);}
}
int main(){pthread_t tid1,tid2;printf("condition variable study!\n");pthread_mutex_init(&mutex,nullptr);pthread_cond_init(&cond,nullptr);pthread_create(&tid1,nullptr,thread1,nullptr);pthread_create(&tid2,nullptr,thread2,nullptr);do{pthread_cond_signal(&cond);}while (1);}
执行片段

这个具体是怎么执行的呢,首先线程1给mutex上锁然后执行pthread_cond_wait,这时候mutex就解锁了,同时线程1或阻塞,所以线程2先打印了applied这句话,然后cond会收到主线程发来的signal信号,这时候线程1就被唤醒了,它也继续执行。
其实最好来说,pthread_mutex_unlock不需要主动调用,因为wait函数本身就会解锁了,而且这样反而有可能会破互斥量的上锁情况。所以可以把unlock那两行注释掉。
clean的那段代码考虑的是上锁但是wait函数被取消的情况,这种情况下会一直上锁,所以让退出回调函数解锁,有备无患。
条件变量都是需要配合互斥量一起使用的,这样可以防止多个线程请求wait函数。
还有一小节是关于异步信号的,不过书上就讲了几行,估计不太重要吧
相关文章:
《Linux C编程实战》笔记:线程同步
这一节主要是解决共享资源的处理。操作系统里也讲过互斥、锁之类的概念。 互斥锁 互斥锁通过锁机制来实现线程同步,同一时刻只允许一个线程执行一个关键部分的代码 一下是操作互斥锁的函数,均声明在pthread.h中。 pthread_mutex_init(初始…...
leetcode141.环形链表
题目 给你一个链表的头节点 head ,判断链表中是否有环。 如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整数 pos 来表示链表尾连接到链表中的位置&#…...
景联文科技:以高质量数据赋能文生图大模型
1月5日,在智求共赢・中国AIGC产业应用峰会暨无界AI生态合作伙伴大会上,中国AIGC产业联盟联合无界AI发布了《中国AIGC文生图产业白皮书2023》,从AIGC文生图发展历程、主流工具、产业实践以及规模预测等多个维度,全面揭示了中国AIGC…...
[论文笔记] PAI-Megatron中qwen和mistral合并到Megtron-LM
一、千问 关于tokenizer的改动: 1.1、更改build_tokenizer中tokenizer类的加载。 /mnt/nas/pretrain/code/Megatron-LM/megatron/tokenizer/__init__.py 或者 tokenizer.py 在build_tokenizer.py函数中: elif args.tokenizer_type == "QwenTokenizer":assert a…...
python设计模式有哪几种
Python 中常见的设计模式有以下几种 一 单例模式(Singleton Pattern):确保一个类只有一个实例,并提供全局访问点。 二 工厂模式(Factory Pattern):使用工厂方法来创建对象,而不是直…...
C语言从入门到实战——数据在内存中的存储方式
数据在内存中的存储方式 前言1. 整数在内存中的存储2. 大小端字节序和字节序判断2.1 什么是大小端2.2 为什么有大小端2.3 练习2.3.1 练习12.3.2 练习22.3.3 练习32.3.4 练习42.3.5 练习52.3.6 练习6 3. 浮点数在内存中的存储3.1 练习3.2 浮点数的存储3.2.1 浮点数存的过程3.2.2…...
高效便捷的远程管理利器——Royal TSX for Mac软件介绍
Royal TSX for Mac是一款功能强大、操作便捷的远程管理软件。无论是远程桌面、SSH、VNC、Telnet还是FTP,用户都可以通过Royal TSX轻松地远程连接和管理各种服务器、计算机和网络设备。 Royal TSX for Mac提供了直观的界面和丰富的功能,让用户能够快速便…...
Docker 部署后端项目自动化脚本
文章目录 开机自启动docker打包后端项目Dockerfile文件脚本文件使用 开机自启动docker systemctl enable dockersystemctl is-enabled docker打包后端项目 这里的项目位置是target同级目录 1.在项目下面新建一个bin目录 新建一个package.txt 写入下方代码后 后缀改为.bat ec…...
MySQL从0到1全教程【2】SQL语言的通用语法及分类
1 SQL语言的通用语法格式 无论是那种数据库的产品,SQL语法都是通用的。 SQL语句可以单行编写也可以多行编写,以分号结尾。SQL语句可以使用空格或者缩进的方式来增强语句的可读性,空格和缩进的数量没有限制。MySQL数据库的SQL语句是不区分大…...
【npm link】Node命令中的npm link命令的使用,还有CLI全局命令的使用,开发命令行工具必不可少的部分
😁 作者简介:一名大四的学生,致力学习前端开发技术 ⭐️个人主页:夜宵饽饽的主页 ❔ 系列专栏:NodeJs 👐学习格言:成功不是终点,失败也并非末日,最重要的是继续前进的勇气…...
Unity组件开发--相机跟随角色和旋转
1.相机跟随组件,节点: 2.相机跟随组件脚本: using System; using System.Collections; using System.Collections.Generic; using Unity.Burst.Intrinsics; using UnityEngine; using UnityEngine.UI;public class CameraFollow : Singleton&…...
JavaScript系列——Proxy(代理)
文章目录 概要Proxy 语法handler 对象的方法Proxy 示例常用handler 对象的方法的参数handler.get()语法示例 handler.set()语法示例 使用场景验证值修正及附加属性 小结 概要 Proxy 用于创建一个对象的代理,将对原对象上的操作(属性获取、赋值、函数调用…...
QT第三天
使用QT完成水果计价界面和功能,如下图: 运行结果: 代码: widget.h #ifndef WIDGET_H #define WIDGET_H#include <QWidget> #include <QListWidgetItem>QT_BEGIN_NAMESPACE namespace Ui { class Widget; } QT_END_N…...
Jetpack Compose -> 声明式UI Modifier
前言 本章主要介绍下 Compose 的声明式 UI 以及初级写法; 什么是声明式UI 传统UI 传统 UI 方式来声明UI <androidx.appcompat.widget.LinearLayoutCompat android:layout_width"match_parent" android:layout_height"match_parent&quo…...
windows10 装docker和docker compose
一.windows环境准备 开启过程中的问题,进入bios修复 二.docker下载安装 1.下载 Docker Desktop: The #1 Containerization Tool for Developers | Docker 下载最新版有问题,下载老版本试试 Docker Desktop release notes | Docker Docs 2.安装 三.do…...
第二次面试总结 - 宏汉科技 - Java后端开发
🧸欢迎来到dream_ready的博客,📜相信您对专栏 “本人真实面经” 很感兴趣o (ˉ▽ˉ;) 专栏 —— 本人真实面经,更多真实面试经验,中大厂面试总结等您挖掘 目录 总结 (非详细) 面试内容(提问内容) - 带答案…...
GPT-4:人工智能的新纪元与未来的无限可能
在人工智能的发展史上,GPT-4的问世标志着一个新的里程碑。作为最新一代的自然语言处理模型,GPT-4不仅在技术上取得了突破,更在应用层面展现了前所未有的潜力。本文将探讨GPT-4的核心技术、应用场景以及它对未来社会的潜在影响。 GPT-4的技术…...
2.右值引用和移动语义
文章目录 右值引用和移动语义&&的特性右值引用优化性能,避免深拷贝移动(move )语义forward 完美转发emplace_back 减少内存拷贝和移动unordered container 无序容器map和unordered_map的差别内部实现机理不同优缺点以及适用处 小结优缺点以及适用处 小结 代…...
深入浅出线程原理
Linux 中的线程本质 线程接口由 Native POSIX Thread Library 提供,即:NPTL 库函数 线程被称为轻量级进程 (Light Weight Process) 每一个线程在内核中都对应一个调度实体,拥有独立的结构体 (task_struct) 内核设计:一个进程对…...
openssl3.2 - 官方demo学习 - saccept.c
文章目录 openssl3.2 - 官方demo学习 - saccept.cEND openssl3.2 - 官方demo学习 - saccept.c 建立TLSServer(使用了证书, 和证书中的私钥), 接收客户端的连接, 并将客户端发来的信息打印到屏幕 笔记 /*! \file saccept.c */ /*! \brief 建立TLSServer(使用了证书, 和证书中…...
自然语言处理——循环神经网络
自然语言处理——循环神经网络 循环神经网络应用到基于机器学习的自然语言处理任务序列到类别同步的序列到序列模式异步的序列到序列模式 参数学习和长程依赖问题基于门控的循环神经网络门控循环单元(GRU)长短期记忆神经网络(LSTM)…...
RNN避坑指南:从数学推导到LSTM/GRU工业级部署实战流程
本文较长,建议点赞收藏,以免遗失。更多AI大模型应用开发学习视频及资料,尽在聚客AI学院。 本文全面剖析RNN核心原理,深入讲解梯度消失/爆炸问题,并通过LSTM/GRU结构实现解决方案,提供时间序列预测和文本生成…...
Xen Server服务器释放磁盘空间
disk.sh #!/bin/bashcd /run/sr-mount/e54f0646-ae11-0457-b64f-eba4673b824c # 全部虚拟机物理磁盘文件存储 a$(ls -l | awk {print $NF} | cut -d. -f1) # 使用中的虚拟机物理磁盘文件 b$(xe vm-disk-list --multiple | grep uuid | awk {print $NF})printf "%s\n"…...
无人机侦测与反制技术的进展与应用
国家电网无人机侦测与反制技术的进展与应用 引言 随着无人机(无人驾驶飞行器,UAV)技术的快速发展,其在商业、娱乐和军事领域的广泛应用带来了新的安全挑战。特别是对于关键基础设施如电力系统,无人机的“黑飞”&…...
C++.OpenGL (20/64)混合(Blending)
混合(Blending) 透明效果核心原理 #mermaid-svg-SWG0UzVfJms7Sm3e {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-SWG0UzVfJms7Sm3e .error-icon{fill:#552222;}#mermaid-svg-SWG0UzVfJms7Sm3e .error-text{fill…...
uniapp 小程序 学习(一)
利用Hbuilder 创建项目 运行到内置浏览器看效果 下载微信小程序 安装到Hbuilder 下载地址 :开发者工具默认安装 设置服务端口号 在Hbuilder中设置微信小程序 配置 找到运行设置,将微信开发者工具放入到Hbuilder中, 打开后出现 如下 bug 解…...
前端中slice和splic的区别
1. slice slice 用于从数组中提取一部分元素,返回一个新的数组。 特点: 不修改原数组:slice 不会改变原数组,而是返回一个新的数组。提取数组的部分:slice 会根据指定的开始索引和结束索引提取数组的一部分。不包含…...
水泥厂自动化升级利器:Devicenet转Modbus rtu协议转换网关
在水泥厂的生产流程中,工业自动化网关起着至关重要的作用,尤其是JH-DVN-RTU疆鸿智能Devicenet转Modbus rtu协议转换网关,为水泥厂实现高效生产与精准控制提供了有力支持。 水泥厂设备众多,其中不少设备采用Devicenet协议。Devicen…...
快速排序算法改进:随机快排-荷兰国旗划分详解
随机快速排序-荷兰国旗划分算法详解 一、基础知识回顾1.1 快速排序简介1.2 荷兰国旗问题 二、随机快排 - 荷兰国旗划分原理2.1 随机化枢轴选择2.2 荷兰国旗划分过程2.3 结合随机快排与荷兰国旗划分 三、代码实现3.1 Python实现3.2 Java实现3.3 C实现 四、性能分析4.1 时间复杂度…...
32单片机——基本定时器
STM32F103有众多的定时器,其中包括2个基本定时器(TIM6和TIM7)、4个通用定时器(TIM2~TIM5)、2个高级控制定时器(TIM1和TIM8),这些定时器彼此完全独立,不共享任何资源 1、定…...
