IO进程 day05
IO进程 day05
- 9. 进程
- 9. 9. 守护进程
- 守护进程的特点
- 守护进程创建步骤
- 10. 线程
- 10.1. 线程的概念
- 10.2. 进程和线程的区别
- 10.2. 线程资源
- 10.3. 线程的函数接口
- 1. pthread_create-创建线程
- 线程函数和普通函数的区别
- 2. pthread_exit
- 3.线程资源回收函数
- join和detach的区别
- 获取线程号(pthread_self)
- 10.4. 线程间的通信机制
- 概念
- 同步机制
- 信号量
- 无名信号量
- 无名信号量函数接口
- 互斥
- 互斥锁
- 互斥锁的函数接口
- 条件变量
- 条件变量函数接口
9. 进程
9. 9. 守护进程
守护进程的特点
- 后台进程
- 生命周期:从系统开始时开启,系统关闭时结束
- 脱离终端控制,周期执行的进程
守护进程创建步骤
1. 创建一个子进程,父进程退出,子进程变成孤儿进程,被init收养,此时变为后台进程(fork)
2. 在子进程中,创建一个新的会话,让子进程成为会话组的组长。为了让子进程完全脱离控制终端(srtsid)
3. 运行路径改为根目录,进程的运行路径不能被删除或卸载,增大子进程的权限(chdir)
4. 重设文件权限掩码,增大进程创建文件时的权限,提高灵活性(umask)
5. 关闭文件描述符,将不需要的文件描述符关闭(close)
示例
#include <stdio.h>
#include <unistd.h>int main()
{pid_t pid = 0;// 创建子进程,父进程退出pid = fork();if(pid == EOF){perror("fork err");return EOF;}if(pid == 0){// 子进程// 创建会话组setsid();// 修改运行路径chdir("/");// 重设文件权限掩码umask(0);// 关闭文件描述符for(int i = 0; i < 3; i++)close(i);// 子进程不结束while(1);}else{exit(0);}
}
练习
练习:创建一个守护进程,循环间隔1s向文件中写入字符串 hello
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>int main(int argc, char const *argv[])
{// 命令行判错if(argc != 2){perror("argc err");return EOF;}pid_t pid = 0;int fd = -1;// 打开文件fd = open(argv[1], O_WRONLY | O_APPEND | O_CREAT, 0666);if(fd == -1){perror("open err");return EOF;}// 创建子进程,父进程退出pid = fork();if(pid == EOF){perror("fork err");return EOF;}if(pid == 0){// 子进程// 创建会话组setsid();// 修改运行路径chdir("/");// 重设文件权限掩码umask(0);// 关闭文件描述符for(int i = 0; i < 3; i++)close(i);// 子进程不结束while(1){sleep(1);write(fd, "hello\n", 6);}}else{exit(0);}
}
10. 线程
10.1. 线程的概念
线程是一个轻量型的进程,为了提高系统的性能引入线程,线程和进程都参与CPU的调度
10.2. 进程和线程的区别
共性:都为操作系统提供并发执行的能力
| 不同点 | 线程 | 进程 |
|---|---|---|
| 资源和调度 | 系统调度的最小单位 | 资源分配的最小单位 |
| 地址空间 | 同一个进程创建的所有 线程共享进程资源 | 地址空间相互独立 |
| 相互通信 | 相对简单 全局变量即可实现 需要考虑临界资源 | 比较复杂 借助进程间的通信机制 |
| 安全性 | 相对较差 进程结束时会导致所有线程退出 | 相对更安全 |
10.2. 线程资源
共享资源:
可执行的指令,静态的数据,进程中打开的文件描述符,信号处理函数。当前的工作目录,用户的ID,用户的组ID
私有资源:
线程ID (TID)、PC(程序计数器)和相关寄存器、堆栈、错误号 (errno)、信号掩码和优先级、执行状态和属性
线程标识:
主线程的 TID 和 PID 是相同的,每个子线程有自己独立的 TID,但它们都共享相同的 PID
10.3. 线程的函数接口
1. pthread_create-创建线程
#include <pthread.h>
int pthread_create(pthread_t *thread, const pthread_attr_t *attr,void *(*start_routine) (void *), void *arg);
功能:创建线程
参数:
pthread_t *thread:线程标识,成功创建线程后,pthread_create 会将新线程的 ID 写入 thread 指向的内存位置。
const pthread_attr_t *attr:线程属性, NULL:代表设置默认属性
void *(*start_routine):函数名,代表线程函数,指向一个函数的指针,这个函数就是线程的执行体(也就是线程的入口函数)。该函数必须符合 void *(*start_routine)(void *) 的原型,即接受一个 void * 类型的参数,并返回一个 void * 类型的值。
void *arg:线程函数传递的参数,不传参为NULL
返回值:成功返回0,失败更新errno
- CPU调度线程也是随机的
- 编译时需要加上
-lpthread
示例:
#include <stdio.h>
#include <pthread.h>
// 线程函数
void *pthreadFun(void *arg)
{printf("子线程结束\n");
}
int main()
{pthread_t tid = 0;// 创建线程if(pthread_create(&tid, NULL, pthreadFun, NULL)){printf("create err\n");return EOF;}printf("主线程执行……\n");sleep(2);printf("主线程执行结束\n");return 0;
}
- 传参
void *pthreadFun(void *arg)
{int num = *((int *)arg);printf("子线程结束\n");
}
int main()
{int a = 121;int *p = &a;
}
线程函数和普通函数的区别
- 普通函数是顺序执行的,手动调用函数,执行函数操作,线程函数在创建的线程中并发执行,不会影响主程序的运行。
- 普通函数同步执行,线程函数异步执行
- 存储位置不一样,普通函数存放在当前线程的栈区空间,线程函数共享进程的全局变量和堆空间。
2. pthread_exit
#include <pthread.h>
void pthread_exit(void *retval);
功能:退出进程
参数:void *retval任意类型数据,一般为NULL
返回值:空
3.线程资源回收函数
#include <pthread.h>
int pthread_join(pthread_t thread, void **retval);
功能:用于等待一个指定的线程结束,阻塞函数
参数:
thread:创建线程对象
void **retval:指针*retval指向线程返回的参数,一般是NULL
返回值:成功返回0,失败errno
#include <pthread.h>
int pthread_detach(pthread_t thread);
功能:让线程结束时自动回收线程资源,让线程和主线程分离
参数:pthread_t thread:线程ID
返回值:成功返回0,失败EOF
join和detach的区别

获取线程号(pthread_self)
#include <pthread.h>
pthread_t pthread_self(void);
功能:获取线程号
参数:无
返回值:成功返回线程ID
10.4. 线程间的通信机制
概念
线程之间是很容易进行通信的,能够通过全局变量实现数据的共享和交换,也就是通过访问临界资源,但是多个线程在同时访问共享数据的对象时需要引入同步和互斥机制。
临界资源:一次只允许一个线程访问的资源叫临界资源
同步机制
同步(synchronization)指的是多个任务(线程)按照约定的顺序相互配合完成一件事情
信号量
Linux中信号量分三类
- 内核信号量:由内核控制路径使用
- Posix信号量
1. 无名信号量:存储在内存中,通常在线程间或父子进程间使用
函数接口:sem_init\sem_wait\sem_post
2. 有名信号量:存储在文件中,在进程间线程间都可以使用
函数接口:sem_open\sem_wait\sem_post\sem_close - System V信号量:是信号量的集合,叫信号灯集,属于IPC对象
函数接口:semget\semctl\semop
无名信号量
信号量:通过信号量实现同步操作,由信号量决定线程是继续运行还是阻塞等待。
信号量代表的是某一类资源,它的值表示系统中该资源的数量,信号量>0的话,表示有资源可以使用,可以申请到资源,继续执行程序,信号量<= 0的话,表示没有资源可以使用,无法申请到资源,阻塞。
信号量是一个受保护的量,只能通过函数接口访问
函数接口
初始化信号量:sem_init()
P操作,申请资源:sem_wait() 资源-1
V操作,释放资源:sem_post() 资源+1
注意:信号量是一个非负的整数,所以一定是 (>=0)
无名信号量函数接口
头文件
#include <semaphore.h>
sem_init()
#include <semaphore.h>
int sem_init(sem_t *sem, int pshared, unsigned int value);
功能:信号量初始化
参数:
sem_t *sem初始化的信号量
int pshared信号量共享范围(0:线程间,1:进程间)
unsigned int value信号量的初值
返回值:成功0,失败EOF
sem_wait()
#include <semaphore.h>
int sem_wait(sem_t *sem)
功能:申请资源,p操作
参数:sem_t *sem信号量
返回值:成功0,失败EOF
注意:函数执行时,信号量大于0,标识有资源可用;信号量为0时,表示没有资源可用,此时程序在此阻塞
sem_post()
#include <semaphore.h>
int sem_post(sem_t *sem);
功能:释放资源,V操作
参数:sem_t *sem信号量
返回值:成功0,失败EOF
sem_getvalue
int sem_getvalue(sem_t *sem, int *sval);
功能:获取信号量的值
参数:
sem_t *sem信号量对象
int *sval信号量的值存放的变量(&)
返回值:成功0,失败errno
sem_destroy
int sem_destroy(sem_t *sem);
功能:销毁信号量
参数:sem_t *sem信号量对象
返回值:成功返回0,失败errno
互斥
互斥:多个线程在访问临界资源时,同一时间只能一个线程进行访问
互斥锁
互斥锁:通过互斥锁可以实现互斥机制,主要用来保护临界资源,每个临界资源都由一个互斥锁来保护,线程必须先获得互斥锁才能访问临界资源,访问完资源后释放该锁。
互斥锁的函数接口
pthread_mutex_init
int pthread_mutex_init(pthread_mutex_t *mutex, pthread_mutexattr_t *attr);
功能:初始化互斥锁
参数:
pthread_mutex_t *mutex互斥锁
pthread_mutexattr_t *attr一般为NULL默认属性
返回值:成功返回0,失败返回EOF
pthread_mutex_lock
int pthread_mutex_lock(pthread_mutex_t *mutex);
功能:申请互斥锁
参数:pthread_mutex_t *mutex互斥锁
返回值:成功返回0,失败返回EOF
pthread_mutex_unlock
int pthread_mutex_unlock(pthread_mutex_t *mutex);
功能:释放互斥锁
参数:pthread_mutex_t *mutex互斥锁
返回值:成功返回0,失败返回EOF
pthread_mutex_destroy
int pthread_mutex_destroy(pthread_mutex_t *mutex);
功能:销毁互斥锁
参数:pthread_mutex_t *mutex互斥锁
返回值:成功返回0,失败返回EOF
条件变量
一般与互斥锁一起用,实现同步机制
条件变量函数接口
pthread_cond_init
int pthread_cond_init(pthread_cond_t *restrict cond,const pthread_condattr_t *restrict attr);
功能:初始化条件变量
参数:
pthread_cond_t *restrict cond条件变量对象
const pthread_condattr_t *restrict attr为NULL,默认变量
返回值:成功:0 失败:非0
pthread_cond_wait
int pthread_cond_wait(pthread_cond_t *restrict cond, pthread_mutex_t *restrict mutex);
功能:等待信号的产生
参数:
pthread_cond_t *restrict cond条件变量对象
pthread_mutex_t *restrict mutex互斥锁
返回值:成功:0 失败:非0
注意:没有信号的时候函数会阻塞,同时解锁;等到条件产生,函数会结束结束阻塞同时上锁
pthread_cond_signal
int pthread_cond_signal(pthread_cond_t *cond);
功能:产生条件变量信号
参数:pthread_cond_t *cond条件变量
返回值:成功:0,失败:非0
注意:一定要pthread_cond_wait先执行,再产生条件
pthread_cond_destroy
int pthread_cond_destroy(pthread_cond_t *cond);
功能:将条件变量销毁
参数:pthread_cond_t *cond条件变量
返回值:成功:0,失败:非0
相关文章:
IO进程 day05
IO进程 day05 9. 进程9. 9. 守护进程守护进程的特点守护进程创建步骤 10. 线程10.1. 线程的概念10.2. 进程和线程的区别10.2. 线程资源10.3. 线程的函数接口1. pthread_create-创建线程线程函数和普通函数的区别 2. pthread_exit3.线程资源回收函数join和detach的区别 获取线程…...
【HeadFirst系列之HeadFirstJava】第6天之认识Java的API:解锁高效开发的钥匙
认识Java的API:解锁高效开发的钥匙 在《Head First Java》的第六章节中,作者深入探讨了Java的API(Application Programming Interface),并强调了它在Java开发中的重要性。Java API 是Java开发工具包(JDK&a…...
4 - AXI GPIO按键控制LED实验
文章目录 1 实验任务2 系统框图3 软件设计 1 实验任务 本实验任务是通过调用PL端AXI GPIO IP核,使用中断机制,实现PL端按键控制 PS端LED的功能。 2 系统框图 3 软件设计 注意事项: AXI GPIO IP核是双沿触发中断,不可设置&…...
题海拾贝:扫雷
Hello大家好!很高兴我们又见面啦!给生活添点passion,开始今天的编程之路! 我的博客:<但凡. 我的专栏:《编程之路》、《数据结构与算法之美》、《题海拾贝》 欢迎点赞,关注! 1、题…...
Deepseek本地部署小实践(c盘)
目录 前言 一、安装ollama 二、打开终端执行run 三、可视化 前言 小鲸鱼出来以后看到很多大佬本地部署AI,自己也想试一试,第一次部署AI,选了一个简单的办法,实践一下,写得比较粗糙。 一、安装ollama 先简单的介绍…...
详细解析d3dx9_27.dll丢失怎么办?如何快速修复d3dx9_27.dll
运行程序时提示“d3dx9_27.dll文件缺失”,通常由DirectX组件损坏或文件丢失引起。此问题可通过系统化修复方法解决,无需重装系统或软件。下文将详细说明具体步骤及注意事项。 一.d3dx9_27.dll缺失问题的本质解析 当系统提示“d3dx9_27.dll丢失”时&…...
【LeetCode刷题之路】leetcode155.最小栈
LeetCode刷题记录 🌐 我的博客主页:iiiiiankor🎯 如果你觉得我的内容对你有帮助,不妨点个赞👍、留个评论✍,或者收藏⭐,让我们一起进步!📝 专栏系列:LeetCode…...
矩阵乘积态简介
定义 矩阵乘积态(Matrix Product State, MPS)是一种用于表示量子多体系统的强大工具,特别是在一维系统中。MPS 是一种张量网络状态,它通过将全局量子态分解为一系列局部张量的乘积来有效地表示量子态。 注释: 量子态表…...
Oracle数据库分区自动删除
说明: 该存储过程部署后,设置成定时任务,每天执行。 每次执行删除partition_position 2的分区,删除之后,partition_position 3的分区会前移到partition_position 为 2; CREATE OR REPLACE PROCEDURE BILL_CENT_JI…...
华三交换机S5560 NQA测试
文章目录 NQA配置介绍实验说明 NQA配置介绍 NQA配置 nqa entry admin testtype icmp-echo //配置NQA类型destination ip 10.1.0.1 //配置探测的目的IPsource ip 10.1.0.2 //配置探测的源IPfrequency 6000 //配置探测的时间history-record enable //历史探测记录…...
Vue全局变量的定义和使用,创建 Store变量、读取、修改
在VUE中,当需要各js、各页面都能读写的全局变量时,可以用store变量,从定义到使用的方法如下 一.定义变量,例:我们定一个全局变量gxh 找到 vue的/ src/ store路径, 在 modules文件夹下创建文件gvar.js 在…...
基于Docker的前端环境管理:从开发环境到生产部署的实现方案
# 基于Docker的前端环境管理:从开发环境到生产部署的实现方案 简介及前端开发环境挑战 简介 是一种容器化平台,可以将应用程序及其依赖项打包为一个容器,提供一种轻量级、可移植的环境。它能够简化开发、部署和运维的流程,提高…...
单片机延时函数怎么写规范?
我们以前在开发产品的时候,肯定会碰到一些延时需求,比如常见的LED闪烁,按键消抖,控制IO口输出时序等等。 别小看延时,这个小问题,想做好,甚至要考虑到程序架构层面。 在开发板上,可能…...
数据结构 1-2 线性表的链式存储-链表
1 原理 顺序表的缺点: 插入和删除移动大量元素数组的大小不好控制占用一大段连续的存储空间,造成很多碎片 链表规避了上述顺序表缺点 逻辑上相邻的两个元素在物理位置上不相邻 头结点 L:头指针 头指针:链表中第一个结点的存储…...
vue2版本elementUI的table分页实现多选逻辑
1. 需求 我们需要在表格页上实现多选要求,该表格支持分页逻辑。 2. 认识属性 表格属性 参数说明类型可选值默认值data显示的数据array——row-key行数据的 Key,用来优化 Table 的渲染;在使用 reserve-selection 功能与显示树形数据时&…...
比特信噪比与信噪比SNR的换算公式
在无线通信系统中,比特信噪比与信噪比(SNR,通常指符号信噪比Es/N0)的换算: 核心公式 E b N 0 SNR R ⋅ log 2 M \boxed{ \frac{E_b}{N_0} \frac{\text{SNR}}{R \cdot \log_2 M} } N0EbR⋅log2MSNR 或…...
设计模式-解释器模式、装饰器模式
解释器模式 定义 给分析对象定义一个语言,并定义语言的文法表示,再设计一个解释器来解释语言中的句子。也就是说,用编译语言的方式来分析应用中的实例。这种模式实现了文法表达式处理的接口,该接口解释一个特定的上下文。 类图 …...
C++初阶——简单实现list
目录 1、前言 2、List.h 3、Test.cpp 1、前言 1. 简单实现std::list,重点:迭代器,类模板,运算符重载。 2. 并不是,所有的类,都需要深拷贝,像迭代器类模板,只是用别的类的资源&am…...
linux 命令+相关配置记录(持续更新...)
linux 命令记录相关配置记录 磁盘切换 cd D:#这里表示切换到D盘查看wsl 安装的linux 子系统 wsl --list -vwsl 卸载 linux 子系统 wsl --unregister -xxx # xxx 表示子系统的名字备份Linux 子系统 导出 wsl --export xxx yyy # xxx 表示子系统的名字 yyy 表示压…...
【PDF预览】使用iframe实现pdf文件预览,加盖章
使用iframe实现pdf文件预览,以及在pdf上添加水印。另外还包括批注、打印、下载、缩放、分页等功能 <iframesrc"http://static.shanhuxueyuan.com/test.pdf"width"100%"height"100%"frameborder"0"></iframe>&l…...
网络运维学习笔记(DeepSeek优化版)002网工初级(HCIA-Datacom与CCNA-EI)子网划分与协议解析
文章目录 子网划分与协议解析1. VLSM与CIDR技术解析1.1 VLSM(Variable Length Subnetwork Mask,可变长子网掩码)1.2 CIDR(Classless Inter-Domain Routing,无类域间路由) 2. 子网划分方法与计算2.1 常规划分…...
在线骑行|基于SpringBoot的在线骑行网站设计与实现(源码+数据库+文档)
在线骑行网站系统 目录 基于SpringBoot的在线骑行设计与实现 一、前言 二、系统设计 三、系统功能设计 5.1用户信息管理 5.2 路线攻略管理 5.3路线类型管理 5.4新闻赛事管理 四、数据库设计 五、核心代码 六、论文参考 七、最新计算机毕设选题推荐 八、源码获取…...
BUUCTF-Web方向21-25wp
目录 [HCTF 2018]admin弱口令session伪造 [MRCTF2020]你传你🐎呢[护网杯 2018]easy_tornado[ZJCTF 2019]NiZhuanSiWei[MRCTF2020]Ez_bypass第一层第二层 [HCTF 2018]admin 打开环境,有三处提示,一个跳转链接,一个登录注册&#x…...
软考——WWW与HTTP
1.万维网(world wide web) 是一个规模巨大的、可以资源互联的资料空间。由URL进行定位,通过HTTP协议传送给使用者,又由HTML来进行文件的展现。 它的主要组成部分是:URL、HTTP、HTML。 (1)URL…...
GO 进行编译时插桩,实现零码注入
Go 编译时插桩 Go 语言的编译时插桩是一种在编译阶段自动注入监控代码的技术,目的是在不修改业务代码的情况下,实现对应用程序的监控和追踪。 基本原理 Go 编译时插桩的核心思想是通过在编译过程中对源代码进行分析和修改,将监控代码注入到…...
为人工智能驱动的交通研究增强路面传感器数据采集
论文标题 英文标题:Enhancing Pavement Sensor Data Harvesting for AI-Driven Transportation Studies 中文标题:为人工智能驱动的交通研究增强路面传感器数据采集 作者信息 Manish Kumar Krishne Gowda Purdue University, 465 Northwestern Avenue,…...
unordered_set和unordered_map的使用
Hello,今天我来为大家介绍一下前几年才刚刚新出的两个容器——unordered_map和unordered_set,这两个容器属于是map系列和set系列中的一种,和map/set不同的是它们的底层,map/set的底层是红黑树,而unordered_map/unorder…...
【实体类】分层设计
【实体类】分层设计 【一】实体类的PO、VO、DO、DAO、BO、DTO、POJO有什么区别【1】PO(Persistent Object)【2】VO(View Object)【3】DO(Domain Object)【4】DAO(Data Access Object)…...
【无人集群系列---无人机集群编队算法】
【无人集群系列---无人机集群编队算法】 一、核心目标二、主流编队控制方法1. 领航-跟随法(Leader-Follower)2. 虚拟结构法(Virtual Structure)3. 行为法(Behavior-Based)4. 人工势场法(Artific…...
C语言基本知识------指针(4)
1. 回调函数是什么? 回调函数就是⼀个通过函数指针调用的函数。 如果你把函数的指针(地址)作为参数传递给另⼀个函数,当这个指针被⽤来调⽤其所指向的函数 时,被调⽤的函数就是回调函数。 void qsort(void base,//指针…...
