Android应用程序组件ContentProvider中的数据更新通知机制和Android系统中的广播(Broadcast)通知机制的实现思路是相似的。在Android的广播机制中,首先是接收者对自己感兴趣的广播进行注册,接着当发送者发出这些广播时,接收者就会得到通知了。然而,ContentProvider中的数据监控机制与Android系统中的广播机制又有三个主要的区别,一是前者是通过URI来把通知的发送者和接收者关联在一起的,而后者是通过Intent来关联的,二是前者的通知注册中心是由ContentService服务来扮演的,而后者是由ActivityManagerService服务来扮演的,三是前者负责接收数据更新通知的类必须要继承ContentObserver类,而后者要继承BroadcastReceiver类。之所以会有这些区别,是由于ContentProivder组件的数据共享功能本身就是建立在URI的基础之上的,因此专门针对URI来设计另外一套通知机制会更实用和方便,而Android系统的广播机制是一种更加通用的事件通知机制,它的适用范围会更广泛一些。
这里我们把ContentProvider的数据更新机制划分为三个单元进行分析,第一个单元是ContentService的启动过程,第二个单元是监控数据变化的ContentObserver的注册过程,第二个单元是数据更新通知的发送过程。
1,ContentService在系统启动的时候就启动起来了,以便后面启动起来的应用程序可以使用它.关于ContentService的启动过程这里没分析了.
2.ContentObserver的注册过程分析
在第二个应用程序中 通过调用ContentResolver对象的registerContentObserver()方法 来注册一个自定义的ContentObserver(MyContentObserver)来监控MyContentProvider这个Content Provider中的数据变化.
从ContentObserver继承下来的子类必须要实现onChange函数。当这个ContentObserver子类负责监控的数据发生变化时,ContentService就会调用它的onChange函数来处理,参数selfChange表示这个变化是否是由自己引起的。在这个应用程序中,MyContentObserver继承了ContentObserver类,它负责监控的URI是"content://com.cj.mycontentprovider/contact".当以这个URI为前缀的URI对应的数据发生改变时,ContentService都会调用这个MyContentObserver类的onChange函数来处理。在MyContentObserver类的onChange函数中,执行的操作就是重新获取MyContentProvider中的数据来更新界面上的联系人信息列表。
在MyContentObserver类的构造函数中,有一个参数handler,它的类型为Handler,它是从MainActivity类的onCreate函数中创建并传过来的。这个handler是用来分发和处理消息用的。由于MainActivity类的onCreate函数是在应用程序的主线程中被调用的,因此,这个handler参数就是和应用程序主线程的消息循环关联在一起的。在后面我们分析数据更新通知的发送过程时,便会看到这个handler参数是如何使用的了。
下面我们就开始分析注册MyContentObserver来监控MyContentProvider中的数据变化的过程.
第一步.ContentResolver.registerContentObserver()
在frameworks/base/core/java/android/content/ContentResolver.java文件中:
public final void registerContentObserver(Uri uri, boolean notifyForDescendents, ContentObserver observer) { registerContentObserver(uri, notifyForDescendents, observer, UserHandle.myUserId()); } /** @hide - designated user version */ public final void registerContentObserver(Uri uri, boolean notifyForDescendents, ContentObserver observer, int userHandle) { try { getContentService().registerContentObserver(uri, notifyForDescendents, observer.getContentObserver(), userHandle); } catch (RemoteException e) { } }
第二步. ContentResolver.getContentService()
在frameworks/base/core/java/android/content/ContentResolver.java文件中:
public static IContentService getContentService() { if (sContentService != null) { return sContentService; } IBinder b = ServiceManager.getService(CONTENT_SERVICE_NAME); ... sContentService = IContentService.Stub.asInterface(b); .. return sContentService; }在ContentResolver类中,有一个静态成员变量sContentService,开始时它的值为null。当ContentResolver类的getContentService函数第一次被调用时,它便会通过ServiceManager类的getService函数来获得前面已经启动起来了的ContentService服务的远程接口,然后把它保存在sContentService变量中。这样,当下次ContentResolver类的getContentService函数再次被调用时,就可以直接把这个ContentService远程接口返回给调用者了。
第三步. ContentResolver.getContentObserver()
在frameworks/base/core/java/android/database/ContentObserver.java文件中:
public abstract class ContentObserver { private final Object mLock = new Object(); private Transport mTransport; // guarded by mLock Handler mHandler; /** * Creates a content observer. * * @param handler The handler to run {@link #onChange} on, or null if none. */ public ContentObserver(Handler handler) { mHandler = handler; } /** * Gets access to the binder transport object. Not for public consumption. * * {@hide} */ public IContentObserver getContentObserver() { synchronized (mLock) { if (mTransport == null) { mTransport = new Transport(this); } return mTransport; } } /**
第四步. ContentService.registerContentObserver()
在frameworks/base/core/java/android/content/ContentService.java文件中:
public void registerContentObserver(Uri uri, boolean notifyForDescendants, IContentObserver observer, int userHandle) { ..... synchronized (mRootNode) { mRootNode.addObserverLocked(uri, observer, notifyForDescendants, mRootNode, Binder.getCallingUid(), Binder.getCallingPid(), userHandle); ... } }它调用了ContentService类的成员变量mRootNode的addObserverLocked函数来注册这个ContentObserver对象observer。成员变量mRootNode的类型为ContentService在内部定义的一个类ObserverNode。
第五步. ObserverNode.addObserverLocked()
在frameworks/base/core/java/android/content/ContentService.java文件中:
public void addObserverLocked(Uri uri, IContentObserver observer, boolean notifyForDescendants, Object observersLock, int uid, int pid, int userHandle) { addObserverLocked(uri, 0, observer, notifyForDescendants, observersLock, uid, pid, userHandle); } private void addObserverLocked(Uri uri, int index, IContentObserver observer, boolean notifyForDescendants, Object observersLock, int uid, int pid, int userHandle) { // If this is the leaf node add the observer if (index == countUriSegments(uri)) {//这里的index 上一步传进来的是0 mObservers.add(new ObserverEntry(observer, notifyForDescendants, observersLock, uid, pid, userHandle)); return; } // Look to see if the proper child already exists String segment = getUriSegment(uri, index); int N = mChildren.size(); for (int i = 0; i < N; i++) { ObserverNode node = mChildren.get(i); if (node.mName.equals(segment)) { node.addObserverLocked(uri, index + 1, observer, notifyForDescendants, observersLock, uid, pid, userHandle); return; } } // No child found, create one ObserverNode node = new ObserverNode(segment); mChildren.add(node); node.addObserverLocked(uri, index + 1, observer, notifyForDescendants, observersLock, uid, pid, userHandle); }
在我们这个情景中,传进来的uri为"content://com.cj.mycontentprovider/contact",从第五步第一个用mRootNode的addObserverLocked()函数来往树上增加一个ObserverNode节点时,传进来的参数index的值为0,而调用countUriSegments("content://com.cj.mycontentprovider/contact")函数的返回值为2,不等于index的值.这里看一下countUriSegments()函数:
private int countUriSegments(Uri uri) { if (uri == null) { return 0; } return uri.getPathSegments().size() + 1; }
--getPathSegments得到uri的path部分,并拆分,去掉"/",取到第一个元素(从第0个开始)。这里只有contact一个元素 .
因此就会往下执行,而通过调用getUriSegment("content://com.cj.mycontentprovider/contact", 0)函数得到的返回值为"com.cj.mycontentprovider"。
private String getUriSegment(Uri uri, int index) { if (uri != null) { if (index == 0) { return uri.getAuthority(); } else { return uri.getPathSegments().get(index - 1); } } else { return null; } }
假设这里是第一次调用树的根节点mRootNode来增加"content://com.cj.mycontentprovider/contact"这个URI,那么在接下来的for循环中,就不会在mRootNode的孩子节点列表mChildren中找到与名称"com.cj.mycontentprovider"对应的ObserverNode,于是就会以"com.cj.mycontentprovider"为名称来创建一个新的ObserverNode,并增加到mRootNode的孩子节点列表mChildren中去,并以这个新的ObserverNode来开始新一轮的addObserverLocked()函数调用。
第二次进入到addObserverLocked()函数时,countUriSegments("content://com.cj.mycontentprovider/contact")的值仍为2(参考前面函数),而index的值为1,因此就会往下执行,这时候通过调用getUriSegment("content://shy.luo.providers.articles/item", 1)函数得到的返回值为"contact"。这里不存在名称为"contact"的孩子节点,于是又会以"contact"为名称来创建一个新的ObserverNode,并以这个新的ObserverNode来开始新一轮的addObserverLocked函数调用。
第三次进入到addObserverLocked()函数时,countUriSegments("content://com.cj.mycontentprovider/contact")的值仍为2,而index的值也为2,因此就会新建一个ObserverEntry对象,并保存在这个以"contact"为名称的ObserverNode的ContentObserver列表mObervers中。
最终我们得到的树形结构如下所示:
mRootNode("")
-- ObserverNode("com.cj.mycontentprovider")
--ObserverNode("contact") , which has a ContentObserver in mObservers
这样,ContentObserver的注册过程就完成了。
ContentValues contentValues = new ContentValues(); contentValues.put("name",name); contentValues.put("number",num); getContentResolver().insert(Uri.parse("content://com.cj.mycontentprovider/contact"),contentValues);
public void notifyChange(Uri uri, ContentObserver observer) { notifyChange(uri, observer, true /* sync to network */); } public void notifyChange(Uri uri, ContentObserver observer, boolean syncToNetwork) { notifyChange(uri, observer, syncToNetwork, UserHandle.getCallingUserId()); } /** * Notify registered observers within the designated user(s) that a row was updated. * * @hide */ public void notifyChange(Uri uri, ContentObserver observer, boolean syncToNetwork, int userHandle) { try { getContentService().notifyChange( uri, observer == null ? null : observer.getContentObserver(), observer != null && observer.deliverSelfNotifications(), syncToNetwork, userHandle); } catch (RemoteException e) { } }
第二步: ContentService.notifyChange()
在frameworks/base/core/java/android/content/ContentService.java文件中
public void notifyChange(Uri uri, IContentObserver observer, boolean observerWantsSelfNotifications, boolean syncToNetwork, int userHandle) { ... // This makes it so that future permission checks will be in the context of this // process rather than the caller's process. We will restore this before returning. try { ArrayList<ObserverCall> calls = new ArrayList<ObserverCall>(); synchronized (mRootNode) { mRootNode.collectObserversLocked(uri, 0, observer, observerWantsSelfNotifications, userHandle, calls); } final int numCalls = calls.size(); for (int i=0; i<numCalls; i++) { ObserverCall oc = calls.get(i); try { oc.mObserver.onChange(oc.mSelfChange, uri); .. } catch (RemoteException ex) { .. } } if (syncToNetwork) { SyncManager syncManager = getSyncManager(); if (syncManager != null) { syncManager.scheduleLocalSync(null /* all accounts */, callingUserHandle, uri.getAuthority()); } } } finally { restoreCallingIdentity(identityToken); } }
第三步: ObserverNode.collectObserversLocked()
在frameworks/base/core/java/android/content/ContentService.java文件中:
public void collectObserversLocked(Uri uri, int index, IContentObserver observer, boolean observerWantsSelfNotifications, int targetUserHandle, ArrayList<ObserverCall> calls) { String segment = null; int segmentCount = countUriSegments(uri); if (index >= segmentCount) { // This is the leaf node, notify all observers collectMyObserversLocked(true, observer, observerWantsSelfNotifications, targetUserHandle, calls); } else if (index < segmentCount){ segment = getUriSegment(uri, index); // Notify any observers at this level who are interested in descendants collectMyObserversLocked(false, observer, observerWantsSelfNotifications, targetUserHandle, calls); } int N = mChildren.size(); for (int i = 0; i < N; i++) { ObserverNode node = mChildren.get(i); if (segment == null || node.mName.equals(segment)) { // We found the child, node.collectObserversLocked(uri, index + 1, observer, observerWantsSelfNotifications, targetUserHandle, calls); if (segment != null) { break; } } } }
private void collectMyObserversLocked(boolean leaf, IContentObserver observer, boolean observerWantsSelfNotifications, int targetUserHandle, ArrayList<ObserverCall> calls) { int N = mObservers.size(); IBinder observerBinder = observer == null ? null : observer.asBinder(); for (int i = 0; i < N; i++) { ObserverEntry entry = mObservers.get(i); // Don't notify the observer if it sent the notification and isn't interested // in self notifications boolean selfChange = (entry.observer.asBinder() == observerBinder); if (selfChange && !observerWantsSelfNotifications) { continue; } // Does this observer match the target user? if (targetUserHandle == UserHandle.USER_ALL || entry.userHandle == UserHandle.USER_ALL || targetUserHandle == entry.userHandle) { // Make sure the observer is interested in the notification if (leaf || (!leaf && entry.notifyForDescendants)) { calls.add(new ObserverCall(this, entry.observer, selfChange)); } } } }第一次调用collectObserversLocked()时,是在mRootNode的这个ObserverNode节点中进行收集ContentObserver的。这时候传进来的uri的值为"content://com.cj.mycontentprovider/contact/d",index的值为0。调用countUriSegments("content://shy.luo.providers.articles/item/n")函数得到的返回值为3(参考前面就知道是3 2+1=3),于是就会调用下面语句:
segment = getUriSegment(uri, 0); // Notify any observers at this level who are interested in descendants collectMyObserversLocked(false, observer, observerWantsSelfNotifications, targetUserHandle, calls);
在接下来的for循环中,在mRootNode的孩子节点列表mChildren中查找名称等于"com.cj.mycontentprovider"的OberverNode节点。在上面分析ContentObserver的注册过程时,我们已经往mRootNode的孩子节点列表mChildren中增加了一个名称为"com.cj.mycontentprovider"的OberverNode节点,因此,这里会成功找到它,并且调用它的collectObserversLocked()函数来继续收集ContentObserver。
第二次进入到collectObserversLocked()函数时,是在名称为"com.cj.mycontentprovider"的OberverNode节点中收集ContentObserver的。这时候传来的uri值不变,但是index的值为1,于是执行下面语句:
segment = getUriSegment(uri, 1); // Notify any observers at this level who are interested in descendants collectMyObserversLocked(false, observer, observerWantsSelfNotifications, targetUserHandle, calls);
</pre><pre name="code" class="java" style="font-size: 14px;">这里得到的segment为"contact"。在我们这个情景中,我们没有在名称为"com.cj.mycontentprovider"的OberverNode节点中注册有ContentObserver,因此这里调用collectMyObserversLocked()函数也不会收集到ContentObserver。
在接下来的for循环中,在名称为"com.cj.mycontentprovider"的ObserverNode节点的孩子节点列表mChildren中查找名称等于"contact"的OberverNode节点。在上面分析ContentObserver的注册过程时,我们已经往名称为"com.cj.mycontentprovider"的ObserverNode节点的孩子节点列表mChildren中增加了一个名称为"contact"的OberverNode节点,因此,这里会成功找到它,并且调用它的collectObserversLocked()函数来继续收集ContentObserver。
第三次进入到collectObserversLocked()函数时,是在名称为"com.cj.mycontentprovider"的OberverNode节点的子节点中名称为"contact"的ObserverNode节点中收集ContentObserver的。这时候传来的uri值不变,但是index的值为2,于是执行下面语句:
segment = getUriSegment(uri, 2); // Notify any observers at this level who are interested in descendants collectMyObserversLocked(false, observer, observerWantsSelfNotifications, targetUserHandle, calls);
返回到第二步中,调用下面语句来通知相应的ContentObserver,它们监控的数据发生变化了:
for (int i=0; i<numCalls; i++) { ObserverCall oc = calls.get(i); try { oc.mObserver.onChange(oc.mSelfChange, uri); } catch (RemoteException ex) { } }
第四步: Transport.onChange()
在frameworks/base/core/java/android/database/ContentObserver.java文件中:
private static final class Transport extends IContentObserver.Stub { private ContentObserver mContentObserver; public Transport(ContentObserver contentObserver) { mContentObserver = contentObserver; } @Override public void onChange(boolean selfChange, Uri uri) { ContentObserver contentObserver = mContentObserver; if (contentObserver != null) { contentObserver.dispatchChange(selfChange, uri); } } public void releaseContentObserver() { mContentObserver = null; } }
第五步:. ContentObserver.dispatchChange()
在frameworks/base/core/java/android/database/ContentObserver.java文件中:
public final void dispatchChange(boolean selfChange, Uri uri) { if (mHandler == null) { onChange(selfChange, uri); } else { mHandler.post(new NotificationRunnable(selfChange, uri)); } }
第六步:. NotificationRunnable.run()
在frameworks/base/core/java/android/database/ContentObserver.java文件中:
private final class NotificationRunnable implements Runnable { private final boolean mSelfChange; private final Uri mUri; public NotificationRunnable(boolean selfChange, Uri uri) { mSelfChange = selfChange; mUri = uri; } @Override public void run() { ContentObserver.this.onChange(mSelfChange, mUri); } }这个函数就直接调用ContentObserver的子类的onChange函数来处理这个数据更新通知了。在我们这个情景中,这个ContentObserver子类便是MyContentObserver了。