24
import java.io.*;
import java.nio.file.*;

public class Tmp {

    public static void main(String [] args) throws IOException {
        int count = 0;
        Path path = Paths.get("C:\\tmp\\");
        WatchService ws = null;
        try {
            ws = FileSystems.getDefault().newWatchService();
            path.register(ws, StandardWatchEventKinds.ENTRY_CREATE, StandardWatchEventKinds.ENTRY_DELETE,
                    StandardWatchEventKinds.ENTRY_MODIFY, StandardWatchEventKinds.OVERFLOW);
        } catch (IOException ioe) {
            ioe.printStackTrace();
        }

        while(true) {
            WatchKey key = null;
            try {
                key = ws.take();
            } catch(InterruptedException ie) {
                ie.printStackTrace();
            }

            for(WatchEvent<?> event: key.pollEvents()) {
                switch(event.kind().name()) {
                    case "OVERFLOW":
                        System.out.println(++count + ": OVERFLOW");
                        break;
                    case "ENTRY_MODIFY":
                        System.out.println(++count + ": File " + event.context() + " is changed!");
                        break;
                    case "ENTRY_CREATE":
                        System.out.println(++count + ": File " + event.context() + " is created!");
                        break;
                    case "ENTRY_DELETE":
                        System.out.println(++count + ": File " + event.context() + " is deleted!");
                        break;
                    default:
                        System.out.println(++count + ": UNKNOWN EVENT!");
                }
            }

            key.reset();
        }    
    }
}

当我运行它然后打开记事本++然后创建一个新的空文件并将其保存a.txtC:\tmp\目录中时,我得到了输出:

1: File a.txt is created!
2: File a.txt is deleted!
3: File a.txt is created!

这是为什么?看起来该文件已创建然后删除然后再次创建。为什么?

当我在文件中放入一些文本并保存时,输出为:

4: File a.txt is changed!
5: File a.txt is changed!

为什么变了两次?

4

4 回答 4

2

Watch Service 的 Modify 事件生成两个事件。当我们修改一个已经存在的文件时,文件系统首先用 0 字节创建它并触发一个修改事件,然后在上面写入数据。然后它再次触发修改事件。这就是为什么它显示了两个修改事件。所以我做了什么来解决这个问题,我只是使用计数器来检查我的任务应该只在偶数计数时触发一次

        Path path = null;
        int count = 0;

        try {
            path = Paths.get(new Utility().getConfDirPath());
            System.out.println("Path: " + path);
        } catch (UnsupportedEncodingException e1) {
            e1.printStackTrace();
        }

        WatchService watchService = null;
        try {
            watchService = FileSystems.getDefault().newWatchService();
            path.register(watchService, StandardWatchEventKinds.ENTRY_MODIFY,StandardWatchEventKinds.ENTRY_DELETE);
        } catch (IOException ioe) {
            ioe.printStackTrace();
        }


        while(true) {
            WatchKey key = null;
            try {
                key = watchService.take();
            } catch(InterruptedException ie) {
                ie.printStackTrace();
            }

            for(WatchEvent<?> event: key.pollEvents()) {
                switch(event.kind().name()) {
                    case "ENTRY_MODIFY":
                        System.out.println(++count + ": File " + event.context() + " is changed!");

                        if (count%2==0) {
                            doOnChange(); // do whatever you want
                        }
                        break;
                    case "ENTRY_DELETE":
                        System.out.println(++count + ": File " + event.context() + " is deleted!");
                        break;
                    default:
                        System.out.println(++count + ": UNKNOWN EVENT!");
                }
            }

            // reset the key
            boolean valid = key.reset();
            if (!valid) {
                System.out.println("Key has been unregistered");
            }

        }    
于 2016-12-22T07:24:03.140 回答
0

文件创建和删除事件在我的系统中正常工作(Window 7 + 1.7.0_21)。

更改事件消息显示该文件上每个Ctrl+s操作的时间数 (n)。

      // Removed the "//" after director name as D://tmp"
      //Added just to see the message with a 1 ms gap.
      Thread.sleep(1000); // after key.reset();

示例:如果我们打开文件并继续按 crtl + s (保存而不进行任何更改/更改)。对于每个保存操作,将(重复)显示以下消息。

     File YourFileName.txt is changed!

原因是在 Windows 中 WatchService 将文件更改与时间戳而不是校验和进行比较。

此处给出的更多描述平台依赖项

于 2013-05-19T14:06:41.963 回答
0

我创建了一个小型 FileWatcher 实用程序库: https ://github.com/finsterwalder/fileutils

它允许设置宽限期。宽限期内的多个事件累积起来,只触发一次。

您不应该在实验中使用 Notepad++,因为您不知道 Notepad++ 在做什么。可能是 Notepad++ 实际上多次写入文件。或者它可以编写一个具有不同名称的文件并在完成时重命名它。编写自己的代码来操作文件并观察它。

于 2017-03-15T19:15:50.280 回答
0

这对我有用

    // get the first event before looping
    WatchKey key = this.watcher.take();

    // reset key (executed twice but not invoke the polled events)
    while (key != null && key.reset() ) { 
      // polled events
      for (final WatchEvent<?> event : key.pollEvents()) {
        System.out.printf("\nGlobal received: %s, event for file: %s\n", event.kind(),
            event.context());

        switch (event.kind().name()) {
        case "ENTRY_CREATE":
          LOGGER.debug("event ENTRY_CREATE");
          break;
        case "ENTRY_DELETE":
          LOGGER.debug("event ENTRY_DELETE");
          break;
        case "ENTRY_MODIFY":
          LOGGER.debug("event ENTRY_MODIFY");
          break;
        default:
          LOGGER.debug("event other [OVERFLOW]");
          break;
        }
      }

      key = this.watcher.take();
    }
于 2016-09-26T13:38:14.167 回答