使用stat来检测文件是否存在(慢?)

时间:2010-07-29 13:25:32

标签: c++ c linux filesystems inotify

我正在使用以下代码来检查文件是否在继续之前已经创建了,事情是文件在文件浏览器中显示的内容很多,然后才被stat检测到...是否存在执行问题此?

//... do something

struct stat buf;

while(stat("myfile.txt", &buf))
  sleep(1);

//... do something else

或者是否有更好的方法来检查文件是否存在?

3 个答案:

答案 0 :(得分:4)

使用inotify,您可以安排内核在发生文件系统更改(例如文件创建)时通知您。这可能是您的文件浏览器用来快速了解文件的原因。

答案 1 :(得分:3)

“stat”系统调用正在收集有关文件的不同信息,例如,指向它的许多硬链接或其“inode”号。您可能希望查看“访问”系统调用,您只能通过在“模式”中指定“F_OK”标志来执行存在检查。

但是,您的代码存在一些问题。每次检查尚不存在的文件时,它会使进程休眠一秒钟。为避免这种情况,您必须使用Jerry Coffin建议的inotify API,以便在您等待的文件存在时通知内核。请记住,如果文件已存在,inotify不会通知您,因此实际上您需要同时使用“access”和“inotify”来避免在创建文件后立即开始查看文件时出现竞争情况。

没有更好或更快的方法来检查文件是否存在。如果你的文件浏览器仍然显示文件的速度比这个程序检测到的要快一些,那么Greg Hewgill关于重命名的想法可能正在发生。

这是一个C ++代码示例,用于设置inotify监视,检查文件是否已存在并等待它:

#include <cstdio>
#include <cstring>
#include <string>

#include <unistd.h>
#include <sys/inotify.h>

int
main ()
{
    const std::string directory = "/tmp";
    const std::string filename = "test.txt";
    const std::string fullpath = directory + "/" + filename;

    int fd = inotify_init ();
    int watch = inotify_add_watch (fd, directory.c_str (),
                                   IN_MODIFY | IN_CREATE | IN_MOVED_TO);

    if (access (fullpath.c_str (), F_OK) == 0)
    {
        printf ("File %s exists.\n", fullpath.c_str ());
        return 0;
    }

    char buf [1024 * (sizeof (inotify_event) + 16)];
    ssize_t length;

    bool isCreated = false;

    while (!isCreated)
    {
        length = read (fd, buf, sizeof (buf));
        if (length < 0)
            break;
        inotify_event *event;
        for (size_t i = 0; i < static_cast<size_t> (length);
             i += sizeof (inotify_event) + event->len)
        {
            event = reinterpret_cast<inotify_event *> (&buf[i]);
            if (event->len > 0 && filename == event->name)
            {
                printf ("The file %s was created.\n", event->name);
                isCreated = true;
                break;
            }
        }
    }

    inotify_rm_watch (fd, watch);
    close (fd);
}

答案 2 :(得分:1)

您的代码将检查文件是否每秒都在那里。您可以使用inotify来获取活动。