ev_loop_fork函数

libev监视器介绍:libev监视器用法-CSDN博客

libev loop对象介绍:loop对象-CSDN博客

libev ev_loop_fork函数介绍:ev_loop_fork函数-CSDN博客

libev API吐血整理:https://download.csdn.net/download/qq_39466755/90794251?spm=1001.2014.3001.5503

用于解决fork函数导致子进程集成的fd集合失效问题

cpp 复制代码
#include <stdio.h>
#include <unistd.h>
#include <sys/event.h>
#include <fcntl.h>

void child_process(int kq) {
    printf("Child: Attempting to use inherited kqueue...\n");
    
    struct kevent events[1];
    int n = kevent(kq, NULL, 0, events, 1, NULL); // 无超时等待
    printf("Child: kevent returned %d events (expected: 1)\n", n);
}

int main() {
    int kq = kqueue();
    int pipe_fd[2];
    pipe(pipe_fd);

    // 监控管道读端
    struct kevent ev;
    EV_SET(&ev, pipe_fd[0], EVFILT_READ, EV_ADD, 0, 0, NULL);
    kevent(kq, &ev, 1, NULL, 0, NULL);

    // 触发事件
    write(pipe_fd[1], "test", 5);

    pid_t pid = fork();
    if (pid == 0) {
        child_process(kq); // 子进程直接使用继承的 kqueue
        _exit(0);
    } else {
        struct kevent events[1];
        int n = kevent(kq, NULL, 0, events, 1, NULL);
        printf("Parent: kevent returned %d events\n", n);
    }
    return 0;
}

运行结果

cpp 复制代码
Child: Attempting to use inherited kqueue...
Child: kevent returned 0 events (expected: 1)  # 子进程事件丢失!
Parent: kevent returned 1 events               # 父进程正常

修改代码子进程可以正常接收父进程的fd集合

cpp 复制代码
#include <ev.h>
#include <stdio.h>
#include <unistd.h>
#include <signal.h>

// 管道读端回调
static void pipe_cb(struct ev_loop *loop, ev_io *w, int revents) {
    char buf[256];
    ssize_t n = read(w->fd, buf, sizeof(buf));
    printf("[%s] Received data: %.*s\n", 
           getpid() == getppid() ? "Parent" : "Child", 
           (int)n, buf);
}

int main() {
    // 忽略 SIGPIPE(防止写入关闭的管道导致进程退出)
    signal(SIGPIPE, SIG_IGN);

    struct ev_loop *loop = EV_DEFAULT;
    int pipe_fd[2];
    pipe(pipe_fd);

    // 监控管道读端
    ev_io pipe_watcher;
    ev_io_init(&pipe_watcher, pipe_cb, pipe_fd[0], EV_READ);
    ev_io_start(loop, &pipe_watcher);

    // 写入数据(触发事件)
    write(pipe_fd[1], "hello", 6);

    pid_t pid = fork();
    if (pid == 0) {
        // ---------- 关键修复 ----------
        ev_loop_fork(loop);  // 重置内核状态
        // ------------------------------
        
        printf("Child: Started event loop\n");
        ev_run(loop, 0);  // 子进程现在能正常接收事件
        _exit(0);
    } else {
        printf("Parent: Started event loop\n");
        ev_run(loop, 0);
    }
    return 0;
}

运行结果

cpp 复制代码
Parent: Started event loop
[Parent] Received data: hello  # 父进程正常接收
Child: Started event loop
[Child] Received data: hello   # 子进程修复后也能接收

结合libev接口,父子进程共享循环时的正确用法

cpp 复制代码
struct ev_loop *loop = EV_DEFAULT;
ev_io parent_watcher;
ev_io_init(&parent_watcher, parent_cb, pipe_fd[0], EV_READ);
ev_io_start(loop, &parent_watcher);

pid_t pid = fork();
if (pid == 0) {
    // 子进程
    ev_loop_fork(loop);  // 先重置后端

    // 添加子进程独有的监视器
    ev_io child_watcher;
    ev_io_init(&child_watcher, child_cb, another_fd, EV_WRITE);
    ev_io_start(loop, &child_watcher);

    ev_run(loop, 0);  // 现在能正确处理父/子监视器的事件
} else {
    // 父进程继续原逻辑
    ev_run(loop, 0);
}

代码解析:

libev 使用底层机制(如 epoll/kqueue)来监听文件描述符。当调用 fork() 时,子进程会继承父进程的 epoll 实例,但该实例可能已失效(内核状态与用户态不一致)。ev_loop_fork() 会重建后端(如重新创建 epoll 实例),确保事件循环在子进程中能正常工作。因为struct ev_loop *loop = EV_DEFAULT;已经创建了底层的事件监听机制(如 epoll、kqueue 或 select 等,具体取决于系统支持)。

即使子进程不直接使用 pipe_fd[0],事件循环本身仍需正确的后端支持。

虽然子进程没有主动使用 parent_watcher(监视 pipe_fd[0]),但该监视器仍存在于 loop 中(因为它是父进程注册的)。未重置的事件循环可能会错误地尝试处理这些继承的监视器,导致未定义行为。

一般情况下都是搭配libev开源库的API函数(ev_fork_init,ev_fork_start等)一起使用:

cpp 复制代码
#include <ev.h>
#include <unistd.h>
#include <stdio.h>

// fork 回调函数
void fork_cb(EV_P_ ev_fork *w, int revents) {
    printf("Child process (PID: %d) reinitializing event loop...\n", getpid());
    ev_loop_fork(EV_A); // 必须调用,重新初始化子进程的事件循环
}

int main() {
    struct ev_loop *loop = EV_DEFAULT;
    struct ev_fork fork_watcher;

    // 初始化fork监视器
    ev_fork_init(&fork_watcher, fork_cb);
    ev_fork_start(loop, &fork_watcher); // 启动监视器

    printf("Parent process (PID: %d) started. Forking...\n", getpid());
    pid_t pid = fork();

    if (pid == 0) {
        // 子进程:ev_loop_fork已在回调中调用
        ev_run(loop, 0); // 子进程事件循环
    } else if (pid > 0) {
        // 父进程代码
        printf("Parent process continues (child PID: %d)\n", pid);
        sleep(2); // 模拟父进程工作
    } else {
        perror("fork failed");
        return 1;
    }

    return 0;
}
相关推荐
C++ 老炮儿的技术栈28 分钟前
C++中什么是函数指针?
c语言·c++·笔记·学习·算法
再睡一夏就好32 分钟前
C语言常见的文件操作函数总结
c语言·开发语言·c++·笔记·学习笔记
我想吃余1 小时前
【Linux修炼手册】Linux开发工具的使用(一):yum与vim
linux·运维·学习·vim
cyy2981 小时前
android 记录应用内存
android·linux·运维
言之。1 小时前
基于 Ubuntu 24.04 部署 WebDAV
linux·运维·ubuntu
朽棘不雕2 小时前
总结C/C++中程序内存区域划分
c语言
前进的程序员2 小时前
ARM 芯片上移植 Ubuntu 操作系统详细步骤
linux·arm开发·ubuntu
程序员JerrySUN2 小时前
驱动开发硬核特训 · Day 30(下篇): 深入解析 lm48100q I2C 音频编解码器驱动模型(基于 i.MX8MP)
linux·驱动开发·架构·音视频
Jtti2 小时前
CentOS服务器中如何解决内存泄漏问题?
linux·服务器·centos
楚灵魈3 小时前
[Linux]从零开始的STM32MP157 Busybox根文件系统构建
linux·arm开发·stm32