linux aio 异步io

linux下的aio有glibc的和内核所提供的,glibc是使用的多线程的模式模拟的,另外一种是真正的内核异步通知了,已经使用在了nginx上,前面看了一下swoole的实现,是类似与glibc那种多线程的模式。不过两种方法都有一定的毛病,多线程模拟自然是有所效率损失,然而内核不能利用系统的缓存,只能以 O_DIRECT 方式做直接 IO,所以看知乎上有一个linux下的异步IO(AIO)是否已成熟?的问题,不过那是2014年的事情了,不知道现在怎么样。

在此之前需要安装好 libaio

sudo apt install libaio-dev

函数

头文件:#include <libaio.h>

例子

一个一步读取文件内容的例子

#include <stdio.h>
#include <fcntl.h>
#include <libaio.h>
#include <malloc.h>
#include <mhash.h>

#define MAX_EVENT 10
#define BUF_LEN 1024

void callback(io_context_t ctx, struct iocb *iocb, long res, long res2) {
    printf("test call\n");
    printf("%s\n", iocb->u.c.buf);
}

int main() {
    int fd = open("/home/huanl/client.ovpn", O_RDONLY, 0);
    io_context_t io_context;
    struct iocb io, *p = &io;
    struct io_event event[MAX_EVENT];
    char *buf = malloc(BUF_LEN);
    memset(buf, 0, BUF_LEN);
    memset(&io_context, 0, sizeof(io_context));

    if (io_setup(10, &io_context)) {
        printf("io_setup error");
        return 0;
    }
    if (fd < 0) {
        printf("open file error");
        return 0;
    }
    io_prep_pread(&io, fd, buf, BUF_LEN, 0);
    io_set_callback(&io, callback);
    if (io_submit(io_context, 1, &p) < 0) {
        printf("io_submit error");
        return 0;
    }

    int num = io_getevents(io_context, 1, MAX_EVENT, event, NULL);
    for (int i = 0; i < num; i++) {
        io_callback_t io_callback = event[i].data;
        io_callback(io_context, event[i].obj, event[i].res, event[i].res2);
    }

    return 0;
}

参考

https://jin-yang.github.io/post/linux-program-aio.html

点赞

发表评论

电子邮件地址不会被公开。 必填项已用*标注

This site uses Akismet to reduce spam. Learn how your comment data is processed.