Linux Inotify详解

本文转自 https://blog.csdn.net/longwang155069/article/details/54016789

描述

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的文件事件对象。

[csharp]  view plain  copy
  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。

 

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

[cpp]  view plain  copy
  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 (*).  

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

 

实例分析

[cpp]  view plain  copy
  1. #include   
  2. #include   
  3. #include   
  4. #include   
  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 \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.  编译代码
[cpp]  view plain  copy
  1. gcc inotify.c -o inotify  

2. 在tmp目录下创建test目录
[cpp]  view plain  copy
  1. mkdir /tmp/test  

3.  检测/tmp/test目录,使用inotify机制
[cpp]  view plain  copy
  1. ./inotify /tmp/test &  

4.  在/tmp/test下创建1.txt文件
[cpp]  view plain  copy
  1. test$ touch /tmp/test/1.txt  
  2. create file: 1.txt  



本文转自: https://blog.csdn.net/longwang155069/article/details/54016789

你可能感兴趣的:(linux)