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)
  • 11. 线程间的通信机制
    • 概念
    • 同步机制
      • 信号量
      • 无名信号量
      • 无名信号量函数接口
    • 互斥
      • 互斥锁
      • 互斥锁的函数接口
    • 条件变量
      • 条件变量函数接口

9. 进程

9. 9. 守护进程

守护进程的特点

  1. 后台进程
  2. 生命周期:从系统开始时开启,系统关闭时结束
  3. 脱离终端控制,周期执行的进程

守护进程创建步骤

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

  1. CPU调度线程也是随机的
  2. 编译时需要加上-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;
}
  1. 传参
void *pthreadFun(void *arg)
{int num = *((int *)arg);printf("子线程结束\n");
}
int main()
{int a = 121;int *p = &a;
}
线程函数和普通函数的区别
  1. 普通函数是顺序执行的,手动调用函数,执行函数操作,线程函数在创建的线程中并发执行,不会影响主程序的运行。
  2. 普通函数同步执行,线程函数异步执行
  3. 存储位置不一样,普通函数存放在当前线程的栈区空间,线程函数共享进程的全局变量和堆空间。

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

11. 线程间的通信机制

概念

 线程之间是很容易进行通信的,能够通过全局变量实现数据的共享和交换,也就是通过访问临界资源,但是多个线程在同时访问共享数据的对象时需要引入同步和互斥机制。
临界资源:一次只允许一个线程访问的资源叫临界资源

同步机制

 同步(synchronization)指的是多个任务(线程)按照约定的顺序相互配合完成一件事情

信号量

Linux中信号量分三类

  1. 内核信号量:由内核控制路径使用
  2. Posix信号量
     1. 无名信号量:存储在内存中,通常在线程间或父子进程间使用
    函数接口:sem_init\sem_wait\sem_post
     2. 有名信号量:存储在文件中,在进程间线程间都可以使用
    函数接口:sem_open\sem_wait\sem_post\sem_close
  3. 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

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

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

相关文章

数字IC低功耗后端设计实现之power gating和isolation技术

考虑低功耗设计需求&#xff0c;下图中间那个功能模块是需要做power domain的&#xff0c;即这个模块需要插MTCMOS。需要开启时&#xff0c;外面的VDD会和这个模块的LOCAL VDD形成通路&#xff0c;否则就是断开即power off状态。 这些低功耗设计实现经验&#xff0c;你真的懂了…

使用 Open3D 批量渲染并导出固定视角点云截图

一、前言 在三维点云处理与可视化中&#xff0c;固定视角批量生成点云渲染截图是一个常见的需求。例如&#xff0c;想要将同一系列的点云&#xff08;PCD 文件&#xff09;在同样的视角下生成序列图片&#xff0c;以便后续合成为视频或进行其他可视化演示。本文将介绍如何使用…

c++的继承

封装、继承和多态是c的三大特性&#xff0c;他们的关系甚为紧密 封装的概念简单易懂&#xff0c;其实就是将数据和操作数据的方法结合在一起&#xff0c;形成一个独立的单元&#xff08;类&#xff09;&#xff0c;通过访问控制符&#xff08;如private、protected和public&…

3dtiles平移旋转工具制作

3dtiles平移旋转缩放原理及可视化工具实现 背景 平时工作中&#xff0c;通过cesium平台来搭建一个演示场景是很常见的事情。一般来说&#xff0c;演示场景不需要多完善的功能&#xff0c;但是需要一批三维模型搭建&#xff0c;如厂房、电力设备、园区等。在实际搭建过程中&…

我是如何从 0 到 1 找到 Web3 工作的?

作者&#xff1a;Lotus的人生实验 关于我花了一个月的时间&#xff0c;从 0 到 1 学习 Web3 相关的知识和编程知识。然后找到了一个 Web3 创业公司实习的远程工作。 &#x1f447;&#x1f447;&#x1f447; 我的背景: 计算机科班&#xff0c;学历还可以(大厂门槛水平) 毕业工…

进程状态(R|S|D|t|T|X|Z)、僵尸进程及孤儿进程

文章目录 一.进程状态进程排队状态&#xff1a;运行、阻塞、挂起 二.Linux下的进程状态R 运行状态&#xff08;running&#xff09;S 睡眠状态&#xff08;sleeping)D 磁盘休眠状态&#xff08;Disk sleep&#xff09;t 停止、暂停状态(tracing stopped)T 停止、暂停状态(stopp…

为什么要将PDF转换为CSV?CSV是Excel吗?

在企业和数据管理的日常工作中&#xff0c;PDF文件和CSV文件承担着各自的任务。PDF通常用于传输和展示静态的文档&#xff0c;而CSV因其简洁、易操作的特性&#xff0c;广泛应用于数据存储和交换。如果需要从PDF中提取、分析或处理数据&#xff0c;转换为CSV格式可能是一个高效…

Starlink卫星动力学系统仿真建模第十讲-基于SMC和四元数的卫星姿态控制示例及Python实现

基于四元数与滑模控制的卫星姿态控制 一、基本原理 1. 四元数姿态表示 四元数运动学方程&#xff1a; 3. 滑模控制设计 二、代码实现&#xff08;Python&#xff09; 1. 四元数运算工具 import numpy as npdef quat_mult(q1, q2):"""四元数乘法""…

CSS—引入方式、选择器、复合选择器、文字控制属性、CSS特性

目录 CSS 1.引入方式 2.选择器 3.复合选择器 4.文字控制属性 5.CSS特性 CSS 层叠样式表&#xff0c;是一种样式表语言&#xff0c;用来描述HTML文档的呈现 书写时一般按照顺序&#xff1a;盒子模型属性—>文字样式—>圆角、阴影等修饰属性 1.引入方式 引入方式方…

OpenHarmony-4.基于dayu800 GPIO 实践(2)

基于dayu800 GPIO 进行开发 1.DAYU800开发板硬件接口 LicheePi 4A 板载 2x10pin 插针&#xff0c;其中有 16 个原生 IO&#xff0c;包括 6 个普通 IO&#xff0c;3 对串口&#xff0c;一个 SPI。TH1520 SOC 具有4个GPIO bank&#xff0c;每个bank最大有32个IO&#xff1a;  …

win11 24h2 远程桌面 频繁断开 已失去连接 2025

一、现象 Windows11自升级2025年2月补丁后版本号为系统版本是26100.3194&#xff0c;远程桌面频繁断开连接&#xff0c;尝试连接&#xff0c;尤其在连接旧的server2012 二、临时解决方案 目前经测试&#xff0c;在组策略中&#xff0c;远程桌面连接客户端&#xff0c;关闭客户…

rust学习笔记6-数组练习704. 二分查找

上次说到rust所有权看看它和其他语言比有什么优势&#xff0c;就以python为例 # Python3 def test():a [1, 3, -4, 7, 9]print(a[4])b a # 所有权没有发生转移del b[4]print(a[4]) # 由于b做了删除&#xff0c;导致a再度访问报数组越界if __name__ __main__:test() 运行结…

Windows安装NVIDIA显卡CUDAD调用GPU,适用于部署deepseek r1

显卡、显卡驱动、CUDA之间的关系 显卡&#xff1a;&#xff08;GPU&#xff09;&#xff0c;主流是NVIDIA的GPU&#xff0c;因为深度学习本身需要大量计算。GPU的并行计算能力&#xff0c;在过去几年里恰当地满足了深度学习的需求。AMD的GPU基本没有什么支持&#xff0c;可以不…

基于无人机遥感的烟株提取和计数研究

一.研究的背景、目的和意义 1.研究背景及意义 烟草作为我国重要的经济作物之一&#xff0c;其种植面积和产量的准确统计对于烟草产业的发展和管理至关重要。传统的人工烟株计数方法存在效率低、误差大、难以覆盖大面积烟田等问题&#xff0c;已无法满足现代烟草种植管理的需求…

《深度学习实战》第3集:循环神经网络(RNN)与序列建模

第3集&#xff1a;循环神经网络&#xff08;RNN&#xff09;与序列建模 引言 在深度学习领域&#xff0c;处理序列数据&#xff08;如文本、语音、时间序列等&#xff09;是一个重要的研究方向。传统的全连接网络和卷积神经网络&#xff08;CNN&#xff09;难以直接捕捉序列中…

【前沿探索篇七】【DeepSeek自动驾驶:端到端决策网络】

第一章 自动驾驶的"感官革命":多模态神经交响乐团 1.1 传感器矩阵的量子纠缠 我们把8路摄像头+4D毫米波雷达+128线激光雷达的融合称为"传感器交响乐",其数据融合公式可以简化为: def sensor_fusion(cam, radar, lidar):# 像素级特征提取 (ResNet-152…

可狱可囚的爬虫系列课程 13:Requests使用代理IP

一、什么是代理 IP 代理 IP&#xff08;Proxy IP&#xff09;是一个充当“中间人”的服务器IP地址&#xff0c;用于代替用户设备&#xff08;如电脑、手机等&#xff09;直接与目标网站或服务通信。用户通过代理IP访问互联网时&#xff0c;目标网站看到的是代理服务器的IP地址&…

https:原理

目录 1.数据的加密 1.1对称加密 1.2非对称加密 2.数据指纹 2.1数据指纹实际的应用 3.数据加密的方式 3.1只使用对称加密 3.2只使用非对称加密 3.3双方都使用对称加密 3.4非对称加密和对称加密一起使用 4.中间人攻击 5.CA证书 5.1什么是CA证书 CA证书的验证 6.https的原理 1.数据…

Github项目管理之 其余分支同步main分支

文章目录 方法&#xff1a;通过 Pull Request 同步分支1. **创建一个从 main 到目标分支的 Pull Request**2. **合并 Pull Request** 注意事项总结 在 GitHub 网页上&#xff0c;你可以通过 Pull Request 的方式将一个分支&#xff08;例如 main 分支&#xff09;的修改同步到…

Aseprite绘画流程案例(5)——花盆

1.最终图片效果 参考素材来源于&#xff1a;手绘像素画第三课&#xff1a;像素画盆花示范&#xff08;无参考图&#xff09;_哔哩哔哩_bilibili 2.流程 1.新建画布40X27的画布&#xff0c;打开显示网格&#xff0c;背景色为白色 2.画出梯形的盆 3.给盆进行亮暗对比上色 4.添…