Linux非阻塞等待示例

news2025/1/16 17:38:19

Linux非阻塞等待实例

  • 非阻塞等待的意义:
  • 简单的多进程编程示例
  • 代码解释

非阻塞等待的意义:

非阻塞等待在多进程编程中的意义主要体现在提高系统的响应性、实现异步任务执行、动态任务管理和多任务协同工作等方面。它允许父进程在等待子进程退出的同时,继续执行其他任务,从而提高系统的并发性和效率,增加系统的灵活性和可扩展性。

简单的多进程编程示例

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>

#define TASK_NUM 5

typedef void (*task_t)();


void download()
{
    printf("this is a download task is rnning!(正在执行下载任务)\n");
}
void printLog()
{
    printf("this is a write log task is rnning!(正在执行日志任务)\n");
}
void show()
{
    printf("this is a show info task is rnning!(正在执行显示信息任务)\n");
}



void initTasks(task_t tasks[], int num)
{
    for(int i = 0; i < num; i++) tasks[i] = NULL;
}

int addTask(task_t tasks[], task_t t)
{
    int i = 0;
    for(; i < TASK_NUM; i++)
    {
        if(tasks[i] == NULL)
        {
            tasks[i] = t;
            return 1;
        }
    }
    return 0;
}

void executeTask(task_t tasks[], int num)
{
    for(int i = 0; i < num; i++)
    {
        if(tasks[i]) tasks[i]();
    }
}

void worker(int cnt)
{
    printf("I am child, pid: %d, cnt: %d\n", getpid(), cnt);
}

int main()
{
    task_t tasks[TASK_NUM];
    initTasks(tasks, TASK_NUM);
    addTask(tasks, download);
    addTask(tasks, printLog);
    addTask(tasks, show);

    pid_t id = fork();
    if(id == 0){
        // child
        int cnt = 10;
        while(cnt)
        {
            worker(cnt);
            sleep(2);
            cnt--;
        }

        exit(0);
    }

    while(1)
    {
        //father
        int status = 0;
        // 非阻塞等待,可以让等待方在返回的时候,顺便做做自己的事情
        pid_t rid = waitpid(id, &status, WNOHANG);
        if(rid > 0)
        {
            // wait success, child quit now;
            printf("child quit success(孩子退出成功), exit code: %d, exit signal: %d\n", (status>>8)&0xFF, status&0x7F);
            break;
        }
        else if(rid == 0)
        {
            printf("##################################################\n");
            // wait success, but child not quit
            printf("child is alive, wait again, father do other thing(孩子还活着,再次等待,父亲做其他事情)....\n");
            // 该函数内部,其实是回调式执行任务
            executeTask(tasks, TASK_NUM); // 也可以在内部进行自己移除&&新增对应的任务
            printf("##################################################\n");
        }
        else{
            // wait failed, child unknow
            printf("wait failed!\n");
            break;
        }

        sleep(1);
    }
    return 0;
}

运行结果:
在这里插入图片描述

在这里插入图片描述

代码解释

这段代码主要使用了 fork() 和 waitpid() 系统调用来创建和管理子进程。下面是代码的详细解释:

包含了必要的头文件:

<stdio.h>:标准输入输出库,用于输入输出操作。
<stdlib.h>:标准库,提供了一些通用工具函数,如内存分配和进程控制。
<unistd.h>:Unix 标准库,提供了对 POSIX 操作系统 API 的访问。
<sys/types.h>:包含了进程标识符 pid_t 的定义。
<sys/wait.h>:包含了等待进程结束的函数 waitpid() 的定义。

定义了一个常量 TASK_NUM,表示任务的数量。
typedef了一个函数指针类型 task_t,用于表示任务函数的类型。

定义了三个具体的任务函数 download()、printLog() 和 show(),用于演示任务的执行过程。这些函数只是简单地打印一些信息。

  • 定义了一个辅助函数 initTasks(),用于初始化任务数组。该函数会将任务数组中的所有元素设置为 NULL。
  • 定义了一个辅助函数 addTask(),用于将任务添加到任务数组中。该函数会遍历任务数组,找到第一个为 NULL 的位置,并将任务函数指针存储在该位置。如果找不到空闲位置,则返回 0;否则返回 1。
  • 定义了一个辅助函数 executeTask(),用于执行任务数组中的任务。该函数会遍历任务数组,对于非空的任务函数指针,调用对应的任务函数。
  • 定义了一个辅助函数 worker(),用于在子进程中执行具体的工作。该函数会打印进程的 ID(使用 getpid() 函数)和计数值。

在 main() 函数中:

  1. 声明了一个任务数组 tasks[TASK_NUM],并通过调用 initTasks() 初始化该数组。
  2. 调用 addTask() 函数将三个任务函数 download()、printLog() 和 show() 添加到任务数组中。
  3. 调用 fork() 创建一个子进程。如果返回值为 0,则表示当前代码在子进程中执行,否则表示在父进程中执行。
  4. 在子进程中,通过一个循环调用 worker() 函数执行具体的工作。每次循环都会打印进程的 ID 和计数值,并使用 sleep(2) 函数暂停 2 秒,然后将计数值减 1,直到计数值为 0。
  5. 在子进程的循环结束后,调用 exit(0) 终止子进程。
  6. 在父进程中,通过一个无限循环调用 waitpid() 函数来等待子进程的退出。waitpid() 函数用于等待指定的子进程结束,并获取子进程的退出状态。
  7. 使用 WNOHANG 选项,使得 waitpid() 在没有子进程退出时立即返回。这样可以让父进程在等待期间执行其他任务。
  8. **如果 waitpid() 返回值大于 0,则表示成功等待到一个子进程退出。**此时打印子进程的退出状态信息,并终止父进程的循环。
  9. 如果 waitpid() 返回值等于 0,则表示当前没有子进程退出,继续执行父进程的其他任务。在这里,调用 executeTask() 函数执行任务数组中的任务,并打印一些信息。
  10. 如果 waitpid() 返回值小于 0,则表示等待失败,打印错误信息,并终止父进程的循环。
  11. 在父进程的循环中,使用 sleep(1) 函数暂停 1秒,以降低循环的频率。

总体来说,这段代码创建了一个子进程,并在父进程中等待子进程的退出。在等待子进程退出的过程中,父进程会执行一些其他任务,并定期执行任务数组中的任务函数。子进程通过循环执行具体的工作,并在完成后退出。在子进程退出后,父进程打印子进程的退出状态信息,并终止循环,结束程序的执行。

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

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

相关文章

【SQL server】 表结构的约束和维护

表结构的约束和维护 修改表结构 (1)添加列 (2)删除列 (3)修改列alter table 表名 add 新列名 数据类型给员工表添加一列邮箱 alter table People add PeopleMail varchar(200)删除列 alter table People drop column PeopleMain修改列 alter table 表名 alter column 列名 数据…

Vulkan渲染引擎开发教程 一、开发环境搭建

一 安装 Vulkan SDK Vulkan SDK 就是我们要搞的图形接口 首先到官网下载SDK并安装 https://vulkan.lunarg.com/sdk/home 二 安装 GLFW 窗口库 GLFW是个跨平台的小型窗口库&#xff0c;也就是显示窗口&#xff0c;图形的载体 去主页下载并安装&#xff0c;https://www.glfw.…

CSS特效014:模仿钟摆效果

CSS常用示例100专栏目录 本专栏记录的是经常使用的CSS示例与技巧&#xff0c;主要包含CSS布局&#xff0c;CSS特效&#xff0c;CSS花边信息三部分内容。其中CSS布局主要是列出一些常用的CSS布局信息点&#xff0c;CSS特效主要是一些动画示例&#xff0c;CSS花边是描述了一些CSS…

测不准原理

测不准原理 算符的对易关系 commutation relation 测不准原理的矢量推导 Schwarz inequality: 设对易关系&#xff1a; 设一个新态&#xff1a; 投影&#xff1a; 那么有&#xff1a; 代回Schwarz inequality 即可证明&#xff1a;

【机器学习算法】机器学习:支持向量机(SVM)

转载自&#xff1a; 【精选】机器学习&#xff1a;支持向量机&#xff08;SVM&#xff09;-CSDN博客 1.概述 1.1&#xff0c;概念 支持向量机&#xff08;SVM&#xff09;是一类按监督学习方式对数据进行二元分类的广义线性分类器&#xff0c;其决策边界是对学习样本求解的最…

java并发编程之基础与原理2

cpu缓存结构剖析 下面说一下概念与作用 CPU缓存即高速缓冲存储器&#xff0c;是位于CPU与主内存间的一种容量较小但速度很高的存储 器。由于CPU的速度远高于主内存&#xff0c;CPU直接从内存中存取数据要等待一定时间周期&#xff0c;Cache中 保存着CPU刚用过或循环使用的一部…

UE 调整材质UV贴图长宽比例

首先&#xff0c;为什么要先减去0.5呢&#xff0c;因为缩放的贴图中心在0,0原点&#xff0c;以这个点缩放效果是这样&#xff1a; 它缩放的图案不会在正中间&#xff0c;因为是以0,0点进行缩放的 以这个图的箭头去缩放图片的&#xff0c;所以不能使得缩放后的图片放在正中心 那…

开源情报 (OSINT)

开源情报 (OSINT)是出于情报目的收集和分析公开数据的行为。 什么是开源数据&#xff1f; 开源数据是公众容易获得或可根据要求提供的任何信息。 OSINT 来源可包括&#xff1a; ▶ 报纸杂志文章以及媒体报道▶ 学术论文和发表的研究▶ 书籍和其他参考资料▶ 社交媒体活动▶…

【DevOps】Git 图文详解(四):Git 使用入门

Git 图文详解&#xff08;四&#xff09;&#xff1a;Git 使用入门 1.创建仓库2.暂存区 add3.提交 commit 记录4.Git 的 “指针” 引用5.提交的唯一标识 id&#xff0c;HEAD~n 是什么意思&#xff1f;6.比较 diff 1.创建仓库 创建本地仓库的方法有两种&#xff1a; 一种是创建…

CMSIS-RTOS在stm32使用

目录&#xff1a; 一、安装和配置CMSIS_RTOS.1.打开KEIL工程&#xff0c;点击MANAGE RUN-TIME Environment图标。2.勾选CMSIS CORE和RTX.3.配置RTOS 时钟频率、任务栈大小和数量&#xff0c; 软件定时器. 二、CMSIS_RTOS内核启动和创建线程。1.包含头文件。2.内核初始化和启动。…

MySQL/Oracle用逗号分割的id怎么实现in (逗号分割的id字符串)。find_in_set(`id`, ‘1,2,3‘) 函数,

1.MySQL 1.1.正确写法 select * from student where find_in_set(s_id, 1,2,3); 1.2.错误示范 select * from student where find_in_set(s_id, 1,2 ,3); -- 注意&#xff0c;中间不能有空格。1、3 select * from student where find_in_set(s_id, 1,2, 3); -- 注意…

快速搜索多个word、excel等文件中内容

如何快速搜索多个word、excel等文件中内容 操作方法 以win11系统为介绍对象。 首先我们打开“我的电脑”-->“文件夹选项”-->“搜索”标签页,在“搜索内容”下方选择&#xff1a;"始终搜索文件名和内容&#xff08;此过程可能需要几分钟&#xff09;"。然后…

浏览器黑暗模式插件

1.Opera浏览器本身黑暗主题 2.Chrome 3.Edge

JRC Monthly Water History, v1.4数据集

简介&#xff1a; JRC Monthly Water History产品&#xff0c;是利用1984至2020年获取的landsat5、landsat7和landsat8的卫星影像&#xff0c;生成的一套30米分辨率的全球地表水覆盖的月度地表水监测地图集。该数据集共有442景数据&#xff0c;包含1984年3月至2020年12月间的月…

Python大数据之linux学习总结——day09_hive调优

hive调优 hive官方配置url: https://cwiki.apache.org/confluence/display/Hive/ConfigurationProperties hive命令和参数配置 hive参数配置的意义: 开发Hive应用/调优时&#xff0c;不可避免地需要设定Hive的参数。设定Hive的参数可以调优HQL代码的执行效率&#xff0c;或帮…

图新地球地图导入操作步骤

1、下载图源&#xff0c;如下&#xff1a; 2、将其全部复制或部分复制&#xff0c;然后回到桌面&#xff0c;打开文件所在位置&#xff0c;如下&#xff1a; 3、将复制的数据粘贴到文件夹下&#xff0c;具体如下&#xff1a; 4、复制到路径如下&#xff1a; 5、复制结果如下&am…

记一次解决Pyqt6/Pyside6添加QTreeView或QTreeWidget导致窗口卡死(未响应)的新路历程,打死我都想不到是这个原因

文章目录 💢 问题 💢🏡 环境 🏡📄 代码💯 解决方案 💯⚓️ 相关链接 ⚓️💢 问题 💢 我在窗口中添加了一个 QTreeWidget控件 ,但是程序在运行期间,只要鼠标进入到 QTreeWidget控件 内进行操作,时间超过几秒中就会出现窗口 未响应卡死的 状态 🏡 环境 �…

HarmonyOS开发Java与ArkTS如何抉择

在“鸿蒙系统实战短视频App 从0到1掌握HarmonyOS”视频课程中&#xff0c;很多学员来问我&#xff0c;在HarmonyOS开发过程中&#xff0c;面对Java与ArkTS&#xff0c;应该选哪样&#xff1f; 本文详细分析Java与ArkTS在HarmonyOS开发过程的区别&#xff0c;力求解答学员的一些…

C/C++数据结构之中缀表达式转换为后缀表达式,删除堆栈元素

在这篇博客中&#xff0c;我们将深入分析一个使用C编写的栈和表达式计算程序。该程序不仅实现了基本的栈操作&#xff0c;还提供了中缀表达式转后缀表达式和删除堆栈中的元素等实用功能。通过逐一讲解每个函数的功能&#xff0c;我们将更全面地理解这个程序的实现。 资源获取&a…

微信(小程序开发): 解决播放音乐没有声音的情况 代码不报错的情况下依旧没有声音的解决方案

解决无声的问题 在此之前&#xff0c;确保代码能够正常执行哈&#xff01;发这个其实没什么&#xff0c;就是有些人光写代码不调试出现了这个问题 其实解决方法特别简单 第一步&#xff1a; 打开项目后&#xff0c;点击三个点&#xff0c;然后选择模拟操作 第二步&#xff…