from : http://blog.csdn.net/u012585964/article/details/52080629
FileObserver只能实现一级目录的监控。
external/toybox/toys/other/inotifyd.c
inotifyd - /sdcard/ 也只能监控一级目录,e.g. 仅能sdcard根目录。
在前面的Linux文件系统Inotify机制 中介绍了Linux对文件变更监控过程。Android系统在此基础上封装了一个FileObserver类来方便使用Inotify机制。FileObserver是一个抽象类,需要定义子类实现该类的onEvent抽象方法,当被监控的文件或者目录发生变更事件时,将回调FileObserver的onEvent()函数来处理文件或目录的变更事件。
事件监控过程
在FileObserver类中定义了一个静态内部类ObserverThread,该线程类才是真正实现文件或目录监控过程。各种类型的FileObserver都拥有一个ObserverThread实例:
frameworks\base\core\Java\android\os\FileObserver.java
- public abstract class FileObserver {
-
- public static final int ALL_EVENTS = ACCESS | MODIFY | ATTRIB | CLOSE_WRITE
- | CLOSE_NOWRITE | OPEN | MOVED_FROM | MOVED_TO | DELETE | CREATE | DELETE_SELF | MOVE_SELF;
-
- private static ObserverThread s_observerThread;
- static {
- s_observerThread = new ObserverThread();
- s_observerThread.start();
- }
-
- private String m_path;
- private Integer m_descriptor;
- private int m_mask;
- }
FileObserver类通过静态方式构造了一个ObserverThread对象:
- public ObserverThread() {
- super("FileObserver");
- m_fd = init();
- }
frameworks\base\core\jni\android_util_FileObserver.cpp
- static jint android_os_fileobserver_init(JNIEnv* env, jobject object)
- {
- #ifdef HAVE_INOTIFY
- return (jint)inotify_init();
- #else
- return -1;
- #endif
- }
inotify_init()函数实现在Linux文件系统Inotify机制 有详细介绍,然后启动ObserverThread线程,ObserverThread线程运行体:
frameworks\base\core\java\android\os\FileObserve$ObserverThread
- public void run() {
- observe(m_fd);
- }
frameworks\base\core\jni\android_util_FileObserver.cpp
- static void android_os_fileobserver_observe(JNIEnv* env, jobject object, jint fd)
- {
- #ifdef HAVE_INOTIFY
- char event_buf[512];
- struct inotify_event* event;
- while (1)
- {
- int event_pos = 0;
-
- int num_bytes = read(fd, event_buf, sizeof(event_buf));
- if (num_bytes < (int)sizeof(*event))
- {
- if (errno == EINTR)
- continue;
- ALOGE("***** ERROR! android_os_fileobserver_observe() got a short event!");
- return;
- }
-
- while (num_bytes >= (int)sizeof(*event))
- {
- int event_size;
- event = (struct inotify_event *)(event_buf + event_pos);
- jstring path = NULL;
- if (event->len > 0)
- {
- path = env->NewStringUTF(event->name);
- }
-
- env->CallVoidMethod(object, method_onEvent, event->wd, event->mask, path);
- if (env->ExceptionCheck()) {
- env->ExceptionDescribe();
- env->ExceptionClear();
- }
- if (path != NULL)
- {
- env->DeleteLocalRef(path);
- }
- event_size = sizeof(*event) + event->len;
- num_bytes -= event_size;
- event_pos += event_size;
- }
- }
- #endif // HAVE_INOTIFY
- }
ObserverThread线程循环从inotify实例句柄中读取事件,然后回调ObserverThread的onEvent函数来处理事件。
frameworks\base\core\java\android\os\FileObserve$ObserverThread
- public void onEvent(int wfd, int mask, String path) {
-
- FileObserver observer = null;
- synchronized (m_observers) {
-
- WeakReference weak = m_observers.get(wfd);
- if (weak != null) {
- observer = (FileObserver) weak.get();
- if (observer == null) {
- m_observers.remove(wfd);
- }
- }
- }
-
- if (observer != null) {
- try {
-
- observer.onEvent(mask, path);
- } catch (Throwable throwable) {
- Log.wtf(LOG_TAG, "Unhandled exception in FileObserver " + observer, throwable);
- }
- }
- }
注册监控watch
FileObserver类提供了startWatching()函数来启动文件监控
frameworks\base\core\java\android\os\FileObserver.java
- public void startWatching() {
- if (m_descriptor < 0) {
- m_descriptor = s_observerThread.startWatching(m_path, m_mask, this);
- }
- }
由ObserverThread线程对象启动监控
frameworks\base\core\java\android\os\FileObserver$ObserverThread
- public int startWatching(String path, int mask, FileObserver observer) {
-
- int wfd = startWatching(m_fd, path, mask);
- Integer i = new Integer(wfd);
- if (wfd >= 0) {
-
- synchronized (m_observers) {
- m_observers.put(i, new WeakReference(observer));
- }
- }
- return i;
- }
ObserverThread又调用native方法android_os_fileobserver_startWatching()来添加一个watch
frameworks\base\core\jni\android_util_FileObserver.cpp
- static jint android_os_fileobserver_startWatching(JNIEnv* env, jobject object, jint fd, jstring pathString, jint mask)
- {
- int res = -1;
- #ifdef HAVE_INOTIFY
- if (fd >= 0)
- {
- const char* path = env->GetStringUTFChars(pathString, NULL);
-
- res = inotify_add_watch(fd, path, mask);
- env->ReleaseStringUTFChars(pathString, path);
- }
- #endif
- return res;
- }
注销监控watch
FileObserver类提供了使用stopWatching()函数来停止文件监控。
frameworks\base\core\java\android\os\FileObserver$ObserverThread
- public void stopWatching() {
- if (m_descriptor >= 0) {
- s_observerThread.stopWatching(m_descriptor);
- m_descriptor = -1;
- }
- }
frameworks\base\core\java\android\os\FileObserve$ObserverThread
- public void stopWatching(int descriptor) {
- stopWatching(m_fd, descriptor);
- }
frameworks\base\core\jni\android_util_FileObserver.cpp
- static void android_os_fileobserver_stopWatching(JNIEnv* env, jobject object, jint fd, jint wfd)
- {
- #ifdef HAVE_INOTIFY
- inotify_rm_watch((int)fd, (uint32_t)wfd);
- #endif // HAVE_INOTIFY
- }