描述

Inotify API用于检测文件系统变化的机制。Inotify可用于检测单个文件,也可以检测整个目录。当检测的对象是一个目录的时候,目录本身和目录里的内容都会成为检测的对象。

此种机制的出现的目的是当内核空间发生某种事件之后,可以立即通知到用户空间。方便用户做出具体的操作。

Inotify API

  • inotify_init(void)

用于创建一个inotify的实例,然后返回inotify事件队列的文件描述符。 同样内核也提供了inotify_init1(int flags)接口函数,当flag等于0的时候,该函数等价于inotify_init(void)函数。

  • inotify_add_watch(int fd, const char* pathname, uint32_t  mask)

该函数用于添加“watch list”,也就是检测列表。 可以是一个新的watch,也可以是一个已经存在的watch。其中fd就是inotify_init的返回值,pathname是要检测目录或者文件的路径,mask就是要检测的事件类型。该函数成功返回的是一个unique的watch描述符。

  • inotify_rm_watch(int fd, int wd)

用于从watch list种移除检测的对象。

 

数据结构

内核使用struct inotify_event代表一个文件事件。当检测的文件对象发生变化时,使用read系统调用就会返回一个或者多个inotify_event的文件事件对象。

 
  1. struct inotify_event {

  2. int wd; /* Watch descriptor */

  3. uint32_t mask; /* Mask of events */

  4. uint32_t cookie; /* Unique cookie associating related

  5. events (for rename(2)) */

  6. uint32_t len; /* Size of name field */

  7. char name[]; /* Optional null-terminated name */

  8. };

.wd:        就是检测的对象的watch descriptor

.mask:    检测事件的mask

.cookie:  和rename事件相关。

.len:        name字段的长度。

.name:    检测对象的name。

可以看到name字段的长度是0,也就是变长的。因为检测的对象的name不定,使用变长可以方便记录检测对象的name。

 

有关检测的事件类型分为好几种,如下:

 
  1. IN_ACCESS File was accessed (read) (*).

  2. IN_ATTRIB Metadata changed, e.g., permissions, timestamps, extended

  3. attributes, link count (since Linux 2.6.25), UID, GID, etc.(*).

  4. IN_CLOSE_WRITE File opened for writing was closed (*).

  5. IN_CLOSE_NOWRITE File not opened for writing was closed (*).

  6. IN_CREATE File/directory created in watched directory (*).

  7. IN_DELETE File/directory deleted from watched directory (*).

  8. IN_DELETE_SELF Watched file/directory was itself deleted.

  9. IN_MODIFY File was modified (*).

  10. IN_MOVE_SELF Watched file/directory was itself moved.

  11. IN_MOVED_FROM File moved out of watched directory (*).

  12. IN_MOVED_TO File moved into watched directory (*).

  13. IN_OPEN File was opened (*).

注释写的很清楚,不再一一解释了。

 

实例分析

 
  1. #include <sys/inotify.h>

  2. #include <unistd.h>

  3. #include <string.h>

  4. #include <stdio.h>

  5.  
  6. /*

  7. struct inotify_event {

  8. int wd; // Watch descriptor

  9. uint32_t mask; // Mask of events

  10. uint32_t cookie; // Unique cookie associating related events (for rename(2))

  11. uint32_t len; // Size of name field

  12. char name[]; // Optional null-terminated name

  13. };

  14.  
  15. */

  16.  
  17. int watch_inotify_events(int fd)

  18. {

  19. char event_buf[512];

  20. int ret;

  21. int event_pos = 0;

  22. int event_size = 0;

  23. struct inotify_event *event;

  24.  
  25. /*读事件是否发生,没有发生就会阻塞*/

  26. ret = read(fd, event_buf, sizeof(event_buf));

  27.  
  28. /*如果read的返回值,小于inotify_event大小出现错误*/

  29. if(ret < (int)sizeof(struct inotify_event))

  30. {

  31. printf("counld not get event!\n");

  32. return -1;

  33. }

  34.  
  35. /*因为read的返回值存在一个或者多个inotify_event对象,需要一个一个取出来处理*/

  36. while( ret >= (int)sizeof(struct inotify_event) )

  37. {

  38. event = (struct inotify_event*)(event_buf + event_pos);

  39. if(event->len)

  40. {

  41. if(event->mask & IN_CREATE)

  42. {

  43. printf("create file: %s\n",event->name);

  44. }

  45. else

  46. {

  47. printf("delete file: %s\n",event->name);

  48. }

  49. }

  50.  
  51. /*event_size就是一个事件的真正大小*/

  52. event_size = sizeof(struct inotify_event) + event->len;

  53. ret -= event_size;

  54. event_pos += event_size;

  55. }

  56.  
  57. return 0;

  58. }

  59.  
  60. int main(int argc, char** argv)

  61. {

  62. int InotifyFd;

  63. int ret;

  64.  
  65. if (argc != 2)

  66. {

  67. printf("Usage: %s <dir>\n", argv[0]);

  68. return -1;

  69. }

  70.  
  71. /*inotify初始化*/

  72. InotifyFd = inotify_init();

  73. if( InotifyFd == -1)

  74. {

  75. printf("inotify_init error!\n");

  76. return -1;

  77. }

  78.  
  79. /*添加watch对象*/

  80. ret = inotify_add_watch(InotifyFd, argv[1], IN_CREATE | IN_DELETE);

  81.  
  82. /*处理事件*/

  83. watch_inotify_events(InotifyFd);

  84.  
  85. /*删除inotify的watch对象*/

  86. if ( inotify_rm_watch(InotifyFd, ret) == -1)

  87. {

  88. printf("notify_rm_watch error!\n");

  89. return -1;

  90. }

  91.  
  92. /*关闭inotify描述符*/

  93. close(InotifyFd);

  94.  
  95. return 0;

  96. }

 

1.  编译代码

gcc inotify.c -o inotify


2. 在tmp目录下创建test目录

mkdir /tmp/test


3.  检测/tmp/test目录,使用inotify机制

./inotify /tmp/test &


4.  在/tmp/test下创建1.txt文件

 
  1. test$ touch /tmp/test/1.txt

  2. create file: 1.txt

Logo

CSDN联合极客时间,共同打造面向开发者的精品内容学习社区,助力成长!

更多推荐