我想测试inotify,所以从互联网上拿了几个例子,修改它以学习各个方面,但失败了,因为它没有像我想的那样工作。首先,我试着查看一个非常好用的目录。
所以我通过一些修改扩展了该文件的示例,但它仅工作一次并在读取功能上被阻止
#include <sys/inotify.h>
#include <unistd.h>
#include <stdio.h>
#define EVENT_SIZE (sizeof (struct inotify_event))
#define BUF_LEN (16 * (EVENT_SIZE + 16))
int main()
{
int fd;
fd = inotify_init();
if (fd < 0)
perror("inotify_init()");
int wd;
wd = inotify_add_watch(fd, "target", IN_CLOSE_WRITE);
if (wd < 0)
perror("inotify_add_watch");
char buf[BUF_LEN];
int len;
while(1) {
len = read(fd, buf, BUF_LEN);
printf("after read\n");
if (len > 0)
{
int i = 0;
while (i < len)
{
struct inotify_event *event;
event = (struct inotify_event *) &buf[i];
printf("wd=%d mask=%x cookie=%u len=%u\n",
event->wd, event->mask,
event->cookie, event->len);
if (event->mask & IN_MODIFY)
printf("file modified %s", event->name);
if (event->len)
printf("name=%s\n", event->name);
i += EVENT_SIZE + event->len;
}
}
}
return 0;
}
所以,我转移到select(),但在这里,它只运行一次,报告两次然后停止报告更改。
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/inotify.h>
#define EVENT_SIZE ( sizeof (struct inotify_event) )
#define BUF_LEN ( 1024 * ( EVENT_SIZE + 16 ) )
int main( int argc, char **argv )
{
int length, i = 0;
int fd;
int wd;
char buffer[BUF_LEN];
struct timeval timeout;
fd = inotify_init();
if ( fd < 0 ) {
perror( "inotify_init" );
}
wd = inotify_add_watch( fd, "target",
IN_CLOSE_WRITE );
fd_set rfds,rfdss;
int ret;
/* zero-out the fd_set */
FD_ZERO (&rfds);
FD_ZERO (&rfdss);
FD_SET (fd, &rfds);
timeout.tv_sec = 5;
timeout.tv_usec = 0;
while(1){
printf("Before select\n");
//rfds = rfdss;
ret = select (fd + 1, &rfds, NULL, NULL, NULL);
printf("After Select\n");
timeout.tv_sec = 5;
timeout.tv_usec = 0;
if (ret < 0)
perror ("select");
else if (!ret){
}
/* timed out! */
else if (FD_ISSET (fd, &rfds)){
printf("file changed============\n");
length = read( fd, buffer, BUF_LEN );
}
}
( void ) inotify_rm_watch( fd, wd );
( void ) close( fd );
exit( 0 );
}
答案 0 :(得分:2)
一些研究表明,受欢迎的编辑以不同的方式保存它。
它们不是直接覆盖该文件,而是实际创建一个临时文件,然后用新的临时文件替换原始文件。所以实际发生的是你实际观看的文件不再存在,因此任何可能会被反映的变化都会被反射回来。
实际遵循此方法的编辑(可能存在更多) Gedit,Geany,vi
直接覆盖文件的编辑器(可能存在更多) 纳米
因此,即使代码是正确的,编辑器的异常行为也可能存在问题
答案 1 :(得分:1)
有关阻止阅读通话,请参阅:https://stackoverflow.com/a/914520/149111
您可能还想从IN_CLOSE_WRITE切换到IN_ALL_EVENTS以确保您没有遗漏某些内容;可能是删除:
最佳做法是观察包含感兴趣的文件的目录而不是单个文件,因为这将占用内核中较少的资源。它还允许您观察“原子”文件替换操作,其中编写器在文件系统上创建一个临时文件(可能在同一个目录中),写入临时文件,然后在最后rename(2)
通过原始文件的顶部。
这种替换方式确保文件的观察者只会注意到文件的完整内容,而不是文件的半写版本。