Android 监控目录文件变化

        公司要求做一个监控系统某一个目录下文件变化,新增或者删除文件变化的功能,查阅了一些资料,现在做一个总结

        android.os包下的FileObserver类是Android提供的一个用于监听文件访问、创建、修改、删除、移动等操作的监听器,基于linux的INotify。

        FileObserver是个抽象类,必须继承它才能使用。每个FileObserver对象监听一个单独的文件或者文件夹,如果监视的是一个文件夹,那么文件夹下所有的文件和一级子目录的改变都会触发监听的事件,但二级子目录变化只会知道有变化,不会知道是具体哪个文件有变化,应该怎么处理后面会说。

监听的事件类型:

(1)ACCESS,即文件被访问。

(2)MODIFY,文件被修改。

(3)ATTRIB,文件属性被修改,如 chmod、chown、touch 等。

(4)CLOSE_WRITE,可写文件被 close。

(5)CLOSE_NOWRITE,不可写文件被 close。

(6)OPEN,文件被 open。

(7)MOVED_FROM,文件被移走,如 mv。

(8)MOVED_TO,文件被移来,如 mv、cp。

(9)CREATE,创建新文件。

(10)DELETE,文件被删除,如 rm。

(11)DELETE_SELF,自删除,即一个可执行文件在执行时删除自己。

(12)MOVE_SELF,自移动,即一个可执行文件在执行时移动自己。

(13)CLOSE,文件被关闭,等同于(IN_CLOSE_WRITE | IN_CLOSE_NOWRITE)。

(14)ALL_EVENTS,包括上面的所有事件。

注:上面的所有静态变量都是FileObserver静态属性。

示例

下面是一个普通的监控目录的例子:

注意,如果直接使用event做判断有些操作事件可能抓不到,所以需要做一下这个处理:
int e = event & FileObserver.ALL_EVENTS;
LogUtil.e("event->e:"+e);
public class FileListener extends FileObserver {

    public EventCallback callback;

    public FileListener(String path) {
        super(path);
        LogUtil.e("进入文件监控");
    }
    public void setEventCallback(EventCallback callback){
        this.callback = callback;
    }
 
    @Override
    public void onEvent(int event, String path) {
        LogUtil.e("pathaaaa:"+path);
        if (null!=path && !"".equals(path)){
            String substring = path.substring(path.lastIndexOf(".")+1);
            LogUtil.e("path:"+path);
            LogUtil.e("substring:"+substring);
            LogUtil.e("event:"+event);

        }else {
            return;
        }

        int e = event & FileObserver.ALL_EVENTS;
        LogUtil.e("event->e:"+e);
        switch (e) {
            case FileObserver.ACCESS:
                Log.e("wannoo", "文件操作___" + e + "__1打开文件后读取文件的操作");
                break;
            case FileObserver.MODIFY:
                Log.e("wannoo", "文件操作___" + e + "__2文件被修改");
                break;
            case FileObserver.ATTRIB:
                Log.e("wannoo", "文件操作___" + e + "__4属性变化");
                break;
            case FileObserver.CLOSE_WRITE:
                Log.e("wannoo", "文件操作___" + e + "__8文件写入或编辑后关闭");
                break;
            case FileObserver.CLOSE_NOWRITE:
                //录音时,最后一个有效回调是这个
                Log.e("wannoo", "文件操作___" + e + "__16只读文件被关闭");

                callback.onEvent(path);


                break;
            case FileObserver.OPEN:
                Log.e("wannoo", "文件操作___" + e + "__32文件被打开");
                break;
            case FileObserver.MOVED_FROM:
                Log.e("wannoo", "文件操作___" + e + "__64移出事件");//试了重命名先MOVED_FROM再MOVED_TO
                break;
            case FileObserver.MOVED_TO:
                Log.e("wannoo", "文件操作___" + e + "__128移入事件");
                break;
            case FileObserver.CREATE:
                Log.e("wannoo", "文件操作___" + e + "__256新建文件");//把文件移动给自己先CREATE在DELETE
                break;
            case FileObserver.DELETE:
                Log.e("wannoo", "文件操作___" + e + "__512有删除文件");//把文件移出去DELETE
                break;
            case FileObserver.DELETE_SELF:
                Log.e("wannoo", "文件操作___" + e + "__1024监听的这个文件夹被删除");
                break;
            case FileObserver.MOVE_SELF:
                Log.e("wannoo", "文件操作___" + e + "__2048监听的这个文件夹被移走");
                break;
            case FileObserver.ALL_EVENTS:
                Log.e("wannoo", "文件操作___" + e + "__4095全部操作");
                break;
        }
    }

     public interface EventCallback{
        void onEvent(String path);
    }

}
其中FileListener(String path)中的path就是你要监控的目录地址      
onEvent(int event, String path)是系统的回调,event是事件类型,path是文件名           

使用的时候直接在oncrate中

fileListener.startWatching();

注销时

fileListener.stopWatching();

还有一点需要注意,FileObserver对象的声明必须是一个全局变量,否则的话监控是不生效的

监控多级目录

上面介绍了监控单级目录的使用,下面来看看多级目录是怎么实现的。

解决方法就是重新实现FileObserver,然后在start的时候去循环为次级目录添加监听。

public class MultiFileObserver extends FileObserver {

    /** Only modification events */
    public static int CHANGES_ONLY = CREATE | MODIFY |DELETE | CLOSE_WRITE
            | DELETE_SELF | MOVE_SELF | MOVED_FROM | MOVED_TO;

    private List<SingleFileObserver> mObservers;
    private String mPath;
    private int mMask;

    public MultiFileObserver(String path) {
        this(path, ALL_EVENTS);
    }

    public MultiFileObserver(String path, int mask) {
        super(path, mask);
        mPath = path;
        mMask = mask;
    }
    @Override
    public void startWatching() {
        if (mObservers != null)
            return;

        mObservers = new ArrayList<SingleFileObserver>();
        Stack<String> stack = new Stack<String>();
        stack.push(mPath);

        while (!stack.isEmpty()) {
            String parent = stack.pop();
            mObservers.add(new SingleFileObserver(parent, mMask));
            File path = new File(parent);
            File[] files = path.listFiles();
            if (null == files)
                continue;
            for (File f : files) {
                if (f.isDirectory() && !f.getName().equals(".")
                        && !f.getName().equals("..")) {
                    stack.push(f.getPath());
                }
            }
        }

        for (int i = 0; i < mObservers.size(); i++) {
            SingleFileObserver sfo =  mObservers.get(i);
            sfo.startWatching();
        }
    };

    @Override
    public void stopWatching() {
        if (mObservers == null)
            return;

        for (int i = 0; i < mObservers.size(); i++) {
            SingleFileObserver sfo = mObservers.get(i);
            sfo.stopWatching();
        }

        mObservers.clear();
        mObservers = null;
    };


    @Override
    public void onEvent(int event, String path) {
        LogUtil.e("pathaaaa:"+path);
        if (null!=path && !"".equals(path)){
            String substring = path.substring(path.lastIndexOf(".")+1);
            LogUtil.e("path:"+path);
            LogUtil.e("substring:"+substring);
            LogUtil.e("event:"+event);

        }else {
            return;
        }
        switch (event) {
            case FileObserver.ACCESS:
                Log.i("RecursiveFileObserver", "ACCESS: " + path);
                break;
            case FileObserver.ATTRIB:
                Log.i("RecursiveFileObserver", "ATTRIB: " + path);
                break;
            case FileObserver.CLOSE_NOWRITE:
                Log.i("RecursiveFileObserver", "CLOSE_NOWRITE: " + path);
                break;
            case FileObserver.CLOSE_WRITE:
                Log.i("RecursiveFileObserver", "CLOSE_WRITE: " + path);
                break;
            case FileObserver.CREATE:
                Log.i("RecursiveFileObserver", "CREATE: " + path);
                break;
            case FileObserver.DELETE:
                Log.i("RecursiveFileObserver", "DELETE: " + path);
                break;
            case FileObserver.DELETE_SELF:
                Log.i("RecursiveFileObserver", "DELETE_SELF: " + path);
                break;
            case FileObserver.MODIFY:
                Log.i("RecursiveFileObserver", "MODIFY: " + path);
                break;
            case FileObserver.MOVE_SELF:
                Log.i("RecursiveFileObserver", "MOVE_SELF: " + path);
                break;
            case FileObserver.MOVED_FROM:
                Log.i("RecursiveFileObserver", "MOVED_FROM: " + path);
                break;
            case FileObserver.MOVED_TO:
                Log.i("RecursiveFileObserver", "MOVED_TO: " + path);
                break;
            case FileObserver.OPEN:
                Log.i("RecursiveFileObserver", "OPEN: " + path);
                break;
            default:
                Log.i("RecursiveFileObserver", "DEFAULT(" + event + " : " + path);
                break;
        }
    }

    /**
     * Monitor single directory and dispatch all events to its parent, with full
     * path.
     */
    class SingleFileObserver extends FileObserver {
        String mPath;

        public SingleFileObserver(String path) {
            this(path, ALL_EVENTS);
            mPath = path;
        }

        public SingleFileObserver(String path, int mask) {
            super(path, mask);
            mPath = path;
        }

        @Override
        public void onEvent(int event, String path) {
            String newPath = mPath + "/" + path;
            MultiFileObserver.this.onEvent(event, newPath);
        }
    }
}

参考:Android--监测文件夹内容变化_超宇的博客-CSDN博客_android监控文件变化

  • 6
    点赞
  • 23
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Android中,你可以使用Java中的WatchService API来监听文件变化。WatchService API可以监控指定目录中的文件文件夹的变化,并在发生变化时触发事件。 以下是一个简单的示例代码,用于监视指定目录中的文件变化: ```java try { // 获取监听器 WatchService watchService = FileSystems.getDefault().newWatchService(); // 创建要监听的目录 Path directory = Paths.get("/path/to/directory"); // 注册监听器 directory.register( watchService, StandardWatchEventKinds.ENTRY_CREATE, StandardWatchEventKinds.ENTRY_MODIFY, StandardWatchEventKinds.ENTRY_DELETE ); // 启动监听循环 while (true) { WatchKey key = watchService.take(); for (WatchEvent<?> event : key.pollEvents()) { // 处理事件 if (event.kind() == StandardWatchEventKinds.ENTRY_CREATE) { System.out.println("文件创建: " + event.context()); } else if (event.kind() == StandardWatchEventKinds.ENTRY_MODIFY) { System.out.println("文件修改: " + event.context()); } else if (event.kind() == StandardWatchEventKinds.ENTRY_DELETE) { System.out.println("文件删除: " + event.context()); } } key.reset(); } } catch (IOException | InterruptedException e) { e.printStackTrace(); } ``` 在这个例子中,我们使用`FileSystems.getDefault().newWatchService()`方法获取WatchService实例。我们使用`Paths.get("/path/to/directory")`方法获取要监听的目录的路径,并使用`directory.register()`方法注册监听器,并指定要监听的事件类型。 在循环中,我们使用`watchService.take()`方法等待事件发生。当事件发生时,我们可以使用`event.kind()`方法获取事件类型,并使用`event.context()`方法获取文件的名称。在此示例中,我们简单地打印出事件类型和文件名称,但你可以根据需要执行其他操作。 请注意,这个示例只能监视指定目录中的文件变化,并且在应用程序关闭时停止监听。如果你需要在应用程序后台持续监听文件变化,你可能需要将这个代码放在一个单独的线程中运行。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值