Inotify機制

描述

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

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/384968.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/384968.shtml
英文地址,請注明出處:http://en.pswp.cn/news/384968.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

文件操作(二進制文件加密解密)

加密 #include<stdio.h> #include<string.h>void code(char *p,size_t n) {size_t i;for(i 0; i < n; i){p[i] 3;} }int main() {FILE *p1 fopen("./a.txt","r");FILE *p2 fopen("./b.txt","w");char buf[1024] {…

北京加密機現場select問題

問題描述 北京項目通過調用我們提供的庫libsigxt.a與加密機通信&#xff0c;c/s架構&#xff0c;客戶端啟用多個線程&#xff0c;每個線程流程有以下三步&#xff0c;連接加密機&#xff0c;簽名&#xff0c;關閉鏈接。在正常運行一段時間后會出現不能連接加密機服務問題。 連…

拼接字符串(帶參程序)

1.用strcat拼接函數可以實現 #include<stdio.h> #include<string.h>int main(int argc,char ** argv) {char str[100] {0};int i;for( i 1; i < argc; i){strcat(str,argv[i]);}printf("str %s\n",str);return 0; } 2.用sprintf函數也可以實現 #in…

詳細解釋signal和sigaction以及SIG_BLOCK

signal&#xff0c;此函數相對簡單一些&#xff0c;給定一個信號&#xff0c;給出信號處理函數則可&#xff0c;當然&#xff0c;函數簡單&#xff0c;其功能也相對簡單許多&#xff0c;簡單給出個函數例子如下&#xff1a; [cpp] view plain copy 1 #include <signal.h>…

處理SIGCHLD信號

在上一講中&#xff0c;我們使用fork函數得到了一個簡單的并發服務器。然而&#xff0c;這樣的程序有一個問題&#xff0c;就是當子進程終止時&#xff0c;會向父進程發送一個SIGCHLD信號&#xff0c;父進程默認忽略&#xff0c;導致子進程變成一個僵尸進程。僵尸進程一定要處理…

文件操作(stat)

/*** stat.c ***/ #include<stdio.h> #include<string.h> #include<sys/stat.h> #include<stdlib.h>int main() {struct stat st {0}; //定義一個結構體&#xff0c;名字叫ststat("./a.txt",&st); //調用完stat函數之后&…

nginx源碼閱讀(一).綜述

前言 nginx作為一款開源的輕量級高性能web服務器,是非常值得立志從事服務端開發方向的人學習的。現今nginx的最新版本是nginx-1.13.6,代碼量也日漸龐大,但是由于其核心思想并沒改變,為了降低閱讀難度,我選擇的是nginx-1.0.15版本,并且由于時間和水平有限,重點關注的是nginx的啟…

文件操作(stat函數)

stat函數可以獲取文件信息 /*** stat.c ***/ #include<stdio.h> #include<string.h> #include<sys/stat.h> #include<stdlib.h>int main() {struct stat st {0}; //定義一個結構體&#xff0c;名字叫ststat("./a.txt",&st); …

文件操作(結構體)

將結構體內容寫入到文件中 #include<stdio.h> #include<string.h>struct student {char name[100];int age; };int main() {struct student st {"wangqinghe",30};FILE * p fopen("./c.txt","wb");fwrite(&st,sizeof(st),1,p…

nginx源碼閱讀(二).初始化:main函數及ngx_init_cycle函數

前言 在分析源碼時,我們可以先把握主干,然后其他部分再挨個分析就行了。接下來我們先看看nginx的main函數干了些什么。 main函數 這里先介紹一些下面會遇到的變量類型: ngx_int_t: typedef intptr_t ngx_int_t; 64位機器上,intptr_t為long int, 即typedef long int intptr_…

EAGAIN、EWOULDBLOCK、EINTR與非阻塞

EWOULDBLOCK&#xff1a;用于非阻塞模式&#xff0c;不需要重新讀或者寫 EINTR&#xff1a;指操作被中斷喚醒&#xff0c;需要重新讀/寫   在Linux環境下開發經常會碰到很多錯誤(設置errno)&#xff0c;其中EAGAIN是其中比較常見的一個錯誤(比如用在非阻塞操作中)。 從字面上…

文件操作(排序)

文本文件&#xff0c;每行代表一個整數&#xff0c;范圍在0~512之間&#xff1b; 要求&#xff1a;對文件排序&#xff0c;不使用堆空間&#xff0c;只使用棧空間。 用srand()和rand()函數生成一定量的隨機數 /*** file.c ***/ #include<stdio.h> #include<string.h&g…

Linux下send錯誤代碼32

問題描述&#xff1a;今天寫程序&#xff0c;socket后send出現這個問題&#xff0c;send的返回值為-1&#xff0c;而errno為32&#xff0c;這個錯誤代碼為broken pipe&#xff0c;即管道破裂。 問題形成原因&#xff1a;后來通過排查研究&#xff0c;發現出現該種問題出現的可能…

系統級性能分析工具perf的介紹與使用

系統級性能優化通常包括兩個階段&#xff1a;性能剖析&#xff08;performance profiling&#xff09;和代碼優化。 性能剖析的目標是尋找性能瓶頸&#xff0c;查找引發性能問題的原因及熱點代碼。 代碼優化的目標是針對具體性能問題而優化代碼或編譯選項&#xff0c;以改善軟…

linux C線程

一個應用程序可以啟動若干個線程&#xff1b;線程&#xff0c;是程序執行的最小單位&#xff1b;一般一個最簡單的程序最少有一個線程&#xff0c;就是程序本身&#xff0c;也是主函數&#xff1b;一個線程阻塞不會影響另一個線程&#xff1b;多線程的進程可以盡可能多的利用系…

fseek函數

fseek函數&#xff1a; int fseek(FILE *_FILE, long _Offset, int _Origin); 函數設置文件指針stream的位置&#xff0c;如果執行成功&#xff0c;stream將指向以fromwhere為基準&#xff0c;偏移量offset&#xff08;指針偏移量&#xff09;個字節的位置&#xff0c;函數返回…

linux ethtool 查看網卡狀態

ethtool 工具關于網絡協商功能介紹&#xff1b; ethtool - Display or change ethernet card settings&#xff08;ethtool 是用來顯示和更改網卡設置的工具&#xff09;&#xff1b;這個工具比較復雜&#xff0c;功能也特別多。由于洋文比較難懂。所以我們還是把網絡設備協商…

ftell函數

ftell函數用于得到文件位置指針當前位置相對于文件首的偏移字節數&#xff0c;在隨機方式存儲文件時&#xff0c;由于文件位置頻繁的前后移動&#xff0c;程序不容易確定文件的當前位置。 /*** a.txt ***/ asd gsdert dfhjtew /*** ftell.c ***/ #include<stdio.h>int ma…

muduo網絡庫源碼閱讀Step by Step

Posted on: Nov 26 2015 Categories: muduo C Tags: muduo 一般寫服務端程序都需要有一個稱手的網絡庫來幫我們處理瑣碎的網絡通信細節&#xff0c;比如連接的建立、關閉&#xff0c;讀取數據&#xff0c;發送數據&#xff0c;接收、發送緩沖區的管理等&#xff0c;常用的C/C網…

C/C++內存問題檢查利器——Purify

C&#xff0f;C內存問題檢查利器——Purify 一、 引言 我們都知道軟件的測試&#xff08;在以產品為主的軟件公司中叫做QA—Quality Assessment&#xff09;占了整個軟件工程的30% -50%&#xff0c;但有這么一種說法&#xff0c;即使是最優秀測試專家設計出來的測試…