函数installProvider执行完成以后,返回到Step 16中的instalContentProviders函数中,执行下面语句:
  1. try{
  2. ActivityManagerNative.getDefault().publishContentProviders(
  3. getApplicationThread(),results);
  4. }catch(RemoteExceptionex){
  5. }
前面已经提到,这个函数调用的作用就是通知ActivityMangerService,需要在这个进程中加载的Content Provider已经完加载完成了,参数results就包含了这些已经加载好的Content Provider接口。
Step 21. ActivityMangerService.publishContentProviders 这个函数定义在frameworks/base/services/java/com/android/server/am/ActivityManagerService.java文件中:
        
  1. publicfinalclassActivityManagerServiceextendsActivityManagerNative
  2. implementsWatchdog.Monitor,BatteryStatsImpl.BatteryCallback{
  3. ......
  4. publicfinalvoidpublishContentProviders(IApplicationThreadcaller,
  5. List<ContentProviderHolder>providers){
  6. ......
  7. synchronized(this){
  8. finalProcessRecordr=getRecordForAppLocked(caller);
  9. ......
  10. finalintN=providers.size();
  11. for(inti=0;i<N;i++){
  12. ContentProviderHoldersrc=providers.get(i);
  13. if(src==null||src.info==null||src.provider==null){
  14. continue;
  15. }
  16. ContentProviderRecorddst=r.pubProviders.get(src.info.name);
  17. if(dst!=null){
  18. mProvidersByClass.put(dst.info.name,dst);
  19. Stringnames[]=dst.info.authority.split(";");
  20. for(intj=0;j<names.length;j++){
  21. mProvidersByName.put(names[j],dst);
  22. }
  23. intNL=mLaunchingProviders.size();
  24. intj;
  25. for(j=0;j<NL;j++){
  26. if(mLaunchingProviders.get(j)==dst){
  27. mLaunchingProviders.remove(j);
  28. j--;
  29. NL--;
  30. }
  31. }
  32. synchronized(dst){
  33. dst.provider=src.provider;
  34. dst.app=r;
  35. dst.notifyAll();
  36. }
  37. ......
  38. }
  39. }
  40. }
  41. }
  42. ......
  43. }
在我们这个情景中,只有一个Content Provider,因此,这里的N等待1。在中间的for循环里面,最重要的是下面这个语句:
  1. ContentProviderRecorddst=r.pubProviders.get(src.info.name);
从这里得到的ContentProviderRecord对象dst,就是在前面Step 7中创建的ContentProviderRecord对象cpr了。在for循环中,首先是把这个Content Provider信息保存好在mProvidersByClass和mProvidersByName中:
  1. mProvidersByClass.put(dst.info.name,dst);
  2. Stringnames[]=dst.info.authority.split(";");
  3. for(intj=0;j<names.length;j++){
  4. mProvidersByName.put(names[j],dst);
  5. }
前面已经说过,这两个Map中,一个是以类名为键值保存Content Provider信息,一个是以authority为键值保存Content Provider信息。 因为这个Content Provider已经加载好了,因此,把它从mLaunchingProviders列表中删除:
  1. intNL=mLaunchingProviders.size();
  2. intj;
  3. for(j=0;j<NL;j++){
  4. if(mLaunchingProviders.get(j)==dst){
  5. mLaunchingProviders.remove(j);
  6. j--;
  7. NL--;
  8. }
  9. }
最后,设置这个ContentProviderRecord对象dst的provider域为从参数传进来的Content Provider远程接口:
  1. synchronized(dst){
  2. dst.provider=src.provider;
  3. dst.app=r;
  4. dst.notifyAll();
  5. }
执行了dst.notiryAll语句后,在Step 7中等待要获取的Content Provider接口加载完毕的线程就被唤醒了。唤醒之后,它检查本地ContentProviderRecord变量cpr的provider域不为null,于是就返回了。它最终返回到Step 5中的ActivityThread类的getProvider函数中,继续往下执行:
  1. IContentProviderprov=installProvider(context,holder.provider,
  2. holder.info,true);
注意,这里是在Article应用程序中进程中执行installProvider函数的,而前面的Step 17的installProvider函数是在ArticlesProvider应用程序进程中执行的。 Step 22. ActivityThread.installProvider 这个函数定义在frameworks/base/core/java/android/app/ActivityThread.java文件中:
  1. publicfinalclassActivityThread{
  2. ......
  3. privatefinalIContentProviderinstallProvider(Contextcontext,
  4. IContentProviderprovider,ProviderInfoinfo,booleannoisy){
  5. ......
  6. if(provider==null){
  7. ......
  8. }elseif(localLOGV){
  9. ......
  10. }
  11. synchronized(mProviderMap){
  12. //Cachethepointerfortheremoteprovider.
  13. Stringnames[]=PATTERN_SEMICOLON.split(info.authority);
  14. for(inti=0;i<names.length;i++){
  15. ProviderClientRecordpr=newProviderClientRecord(names[i],provider,
  16. localProvider);
  17. try{
  18. provider.asBinder().linkToDeath(pr,0);
  19. mProviderMap.put(names[i],pr);
  20. }catch(RemoteExceptione){
  21. returnnull;
  22. }
  23. }
  24. ......
  25. }
  26. returnprovider;
  27. }
  28. ......
  29. }
同样是执行installProvider函数,与Step 17不同,这里传进来的参数provider是不为null的,因此,它不需要执行在本地加载Content Provider的工作,只需要把从ActivityMangerService中获得的Content Provider接口保存在成员变量mProviderMap中就可以了。 这样,获取与"shy.luo.providers.artilces"这个uri对应的Content Provider(shy.luo.providers.articles.ArticlesProvider)就完成了,它同时也是启动Content Provider的完整过程。第三方应用程序获得了这个Content Provider的接口之后,就可以访问它里面的共享数据了。在下面一篇文章中,我们将重点分析Android应用程序组件Content Provider在不同进程中传输数据的过程,即Content Provider在不同应用程序中共享数据的原理,敬请关注。

更多相关文章

  1. android adapter的性能小结
  2. [Android]加载外部.9图片
  3. 修改android 开机之后的默认语言成中文
  4. Android通过反射实现强制停止应用程序的方法
  5. Android加载网络图片
  6. Android(安卓)app启动流程(Activity启动过程)【基于API26】
  7. Android(安卓)bluetooth介绍(四): a2dp connect流程分析
  8. Android数据库编程:SqLiteOpenHelper的使用

随机推荐

  1. Android(安卓)屏幕旋转 处理 AsyncTask
  2. android 按钮
  3. Windows系统下的Android开发环境搭建
  4. What Android(安卓)Can Learn From the i
  5. Android单元测试遇到的问题
  6. Android入门第十二篇之Gallery
  7. Android获取应用程序的版本名称和版本号
  8. Linux 下Android(安卓)开发环境搭建 ---Cent
  9. android大图片显示
  10. Android中图像变换Matrix的原理、代码验