文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

监听风云之一 Inotify 介绍

2024-12-03 05:47

关注

本文转载自微信公众号「Linux内核那些事」,作者songsong001。转载本文请联系Linux内核那些事公众号。

不知道大家用过 Dropbox 没有,这是国外一款非常好用云盘,你只需要在 Dropbox 中设置好要同步的目录,每当此目录中的文件发生变动时,Dropbox 就会自动把文件同步到云端。

那么,Dorpbox 是怎么知道目录的文件发生了改变呢?答案是,通过 inotfiy 这个系统功能来实现的。

我们主要分为两篇文章来介绍 inotify 这个功能:本篇首先介绍 inotify 的使用方式,而下篇主要介绍 inotify 的实现原理。

inotify 接口们

其实 inotify 的接口比较少,只有3个:inotify_init、inotify_add_watch 和 inotify_rm_watch。下面我们介绍一下这三个接口的作用和原型。

1. inotify_init

inotify_init 函数用于创建一个 inotify 的句柄,可以认为此句柄就是 inotify 的对象。其原型如下:

  1. int inotify_init(void); 

2. inotify_add_watch

创建好 inotify 句柄后,就可以通过调用 inotify_add_watch 函数添加要进行监听的文件或者目录。其原型如下:

  1. int inotify_add_watch(int fd, const char *path, uint32_t mask); 

inotify_add_watch 调用成功后,会返回被监听文件或目录的描述符。下面介绍一下各个参数的意义:

类型 描述
IN_ACCESS 文件被访问
IN_ATTRIB 文件元数据改变
IN_CLOSE_WRITE 关闭为了写入而打开的文件
IN_CLOSE_NOWRITE 关闭只读方式打开的文件
IN_CREATE 在监听目录内创建了文件/目录
IN_DELETE 在监听目录内删除文件/目录
IN_DELETE_SELF 监听目录/文件本身被删除。
IN_MODIFY 文件被修改
IN_MOVE_SELF 受监控目录/文件本身被移动
IN_MOVED 文件被移
IN_OPEN 文件被打开
IN_ALL_EVENTS 以上所有输出事件的统称

3. inotify_rm_watch

inotify_rm_watch 函数用于删除被监听的文件或目录,其原型如下:

  1. int inotify_rm_watch(int fd, uint32_t wd); 

下面介绍一下各个参数的意义:

读取变动事件

介绍完 inotify 的接口后,现在通过一个简单的例子来展示怎么使用 inotify。在编写 inotify 的实例前,先介绍一下怎么获取被监听文件或目录的变动事件。inotify 并没有提供特定的接口来获取被监听的文件或目录的变动事件,而是通过通用的 read 函数来读取,我们来看看 read 函数的原型:

  1. int read(int fd, void *events, size_t len); 

下面说说各个参数的意义:

events 参数用于存放被监听文件或目录的变动事件,一般指定为 inotify_event 结构的数组,inotify_event 结构的定义如下:

  1. struct inotify_event { 
  2.    int         wd;      // 被监控文件或目录的描述符(由inotify_add_watch) 
  3.    uint32_t    mask;    // 变动的事件 
  4.    uint32_t    cookie;  // 比较少使用,可以忽略 
  5.    uint32_t    len;     // name的长度 
  6.    char        name[];  // 用于存放发生变动的文件或目录名称 
  7. }; 

使用实例

现在我们可以使用 inotify 来编写实例了,这个实例主要介绍怎么使用 inotify 监听一个文件或者目录,并且打印其变动事件。

实现代码如下:

  1. #include  
  2. #include  
  3. #include  
  4. #include   // 引入 inotify 的头文件 
  5.  
  6.  
  7. void display_event(const char *base, struct inotify_event *event) 
  8.    char *operate; 
  9.    int mask = event->mask; 
  10.  
  11.    if (mask & IN_ACCESS)        operate = "ACCESS"
  12.    if (mask & IN_ATTRIB)        operate = "ATTRIB"
  13.    if (mask & IN_CLOSE_WRITE)   operate = "CLOSE_WRITE"
  14.    if (mask & IN_CLOSE_NOWRITE) operate = "CLOSE_NOWRITE"
  15.    if (mask & IN_CREATE)        operate = "CREATE"
  16.    if (mask & IN_DELETE_SELF)   operate = "DELETE_SELF"
  17.    if (mask & IN_MODIFY)        operate = "MODIFY"
  18.    if (mask & IN_MOVE_SELF)     operate = "MOVE_SELF"
  19.    if (mask & IN_MOVED_FROM)    operate = "MOVED_FROM"
  20.    if (mask & IN_MOVED_TO)      operate = "MOVED_TO"
  21.    if (mask & IN_OPEN)          operate = "OPEN"
  22.    if (mask & IN_IGNORED)       operate = "IGNORED"
  23.    if (mask & IN_DELETE)        operate = "DELETE"
  24.    if (mask & IN_UNMOUNT)       operate = "UNMOUNT"
  25.  
  26.    printf("%s/%s: %s\n", base, event->name, operate); 
  27.  
  28. #define EVENTS_BUF_SIZE 4096 
  29.  
  30. int main(int argc, char const *argv[]) 
  31.    int fd; 
  32.    int nbytes, offset; 
  33.    char events[EVENTS_BUF_SIZE]; 
  34.    struct inotify_event *event; 
  35.  
  36.    fd = inotify_init(); // 创建 inotify 句柄 
  37.    if (fd < 0) { 
  38.        printf("Failed to initalize inotify\n"); 
  39.        return -1; 
  40.   } 
  41.  
  42.    // 从命令行参数获取要监听的文件或目录路径 
  43.    // 添加要监听的文件或者目录, 监听所有事件 
  44.    if (inotify_add_watch(fd, argv[1], IN_ALL_EVENTS) == -1) { 
  45.        printf("Failed to add file or directory watch\n"); 
  46.        return -1; 
  47.   } 
  48.  
  49.    for (;;) { 
  50.        memset(events, 0, sizeof(events)); 
  51.  
  52.        // 读取发生的事件 
  53.        nbytes = read(fd, events, sizeof(events)); 
  54.        if (nbytes <= 0) { 
  55.            printf("Failed to read events\n"); 
  56.            continue
  57.       } 
  58.  
  59.        // 开始打印发生的事件 
  60.        for (offset = 0; offset < nbytes; ) { 
  61.            event = (struct inotify_event *)&events[offset]; // 获取变动事件的指针 
  62.  
  63.            display_event(argv[1], event); 
  64.  
  65.            offset += sizeof(struct inotify_event) + event->len; // 获取下一个变动事件的偏移量 
  66.       } 
  67.   } 
  68.  
  69.    return 0; 

上面的实例逻辑比较简单,主要步骤如下:

上面实例比较难懂的就是从 events 参数中获取变动事件的指针,我们可以通过下面这幅图来理清获取变动事件指针的逻辑:

通过上图,就比较容易理解怎么从 events 缓冲区中获取到变动事件的指针了。

最后,来看看我们编写的实例的效果动画:

总结

 

本文主要介绍 inotify 的使用,在下一篇文章中,我们将会介绍 inotify 的原理和实现,敬请期待(当然对 inotify 的实现没兴趣的就不用期待了...)。

 

来源: Linux内核那些事内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯