Android应用程序组件Content Provider的启动过程源代码分析(4)

简介:
  接下去这个代码判断当前要获取的Content Provider是否允许在客户进程中加载,即查看一个这个Content Provider否配置了multiprocess属性为true,如果允许在客户进程中加载,就直接返回了这个Content Provider的信息了:
 
 
  1. if (r != null && cpr.canRunHere(r)) {   
  2.     // If this is a multiprocess provider, then just return its   
  3.     // info and allow the caller to instantiate it.  Only do   
  4.     // this if the provider is the same user as the caller's   
  5.     // process, or can run as root (so can be in any process).   
  6.     return cpr;   
  7. }   
 在我们这个情景中,要获取的ArticlesProvider设置了要在独立的进程中运行,因此,继续往下执行: 
 
 
  1. // This is single process, and our app is now connecting to it.   
  2. // See if we are already in the process of launching this   
  3. // provider.   
  4. final int N = mLaunchingProviders.size();   
  5. int i;   
  6. for (i=0; i<N; i++) {   
  7.     if (mLaunchingProviders.get(i) == cpr) {   
  8.         break;   
  9.     }   
  10. }   
   系统中所有正在加载的Content Provider都保存在mLaunchingProviders成员变量中。在加载相应的Content Provider之前,首先要判断一下它是可否正在被其它应用程序加载,如果是的话,就不用重复加载了。在我们这个情景中,没有其它应用程序也正在加载ArticlesProvider这个Content Provider,继续往前执行:
  1. // If the provider is not already being launched, then get it  
  2. // started.  
  3. if (i >= N) {  
  4.     final long origId = Binder.clearCallingIdentity();  
  5.     ProcessRecord proc = startProcessLocked(cpi.processName,  
  6.         cpr.appInfo, false0"content provider",  
  7.         new ComponentName(cpi.applicationInfo.packageName,  
  8.         cpi.name), false);  
  9.     ......  
  10.     mLaunchingProviders.add(cpr);  
  11.     ......  
  12. }  
        这里的条件i >= N为true,就表明没有其它应用程序正在加载这个Content Provider,因此,就要调用startProcessLocked函数来启动一个新的进程来加载这个Content Provider对应的类了,然后把这个正在加载的信息增加到mLaunchingProviders中去。我们先接着分析这个函数,然后再来看在新进程中加载Content Provider的过程,继续往下执行:
  1. // Make sure the provider is published (the same provider class  
  2. // may be published under multiple names).  
  3. if (firstClass) {  
  4.     mProvidersByClass.put(cpi.name, cpr);  
  5. }  
  6. cpr.launchingApp = proc;  
  7. mProvidersByName.put(name, cpr);  
        这段代码把这个Content Provider的信息分别保存到mProvidersByName和mProviderByCalss两个Map中去,以方便后续查询。
 
        因为我们需要获取的Content Provider是在新的进程中加载的,而getContentProviderImpl这个函数是在系统进程中执行的,它必须要等到要获取的Content Provider是在新的进程中加载完成后才能返回,这样就涉及到进程同步的问题了。这里使用的同步方法是不断地去检查变量cpr的provider域是否被设置了。当要获取的Content Provider在新的进程加载完成之后,它会通过Binder进程间通信机制调用到系统进程中,把这个cpr变量的provider域设置为已经加载好的Content Provider接口,这时候,函数getContentProviderImpl就可以返回了。下面的代码就是用来等待要获取的Content Provider是在新的进程中加载完成的:
  1. // Wait for the provider to be published...  
  2. synchronized (cpr) {  
  3.     while (cpr.provider == null) {  
  4.         ......  
  5.         try {  
  6.             cpr.wait();  
  7.         } catch (InterruptedException ex) {  
  8.         }  
  9.     }  
  10. }  
        下面我们再分析在新进程中加载ArticlesProvider这个Content Provider的过程。
 
         Step 8. ActivityManagerService.startProcessLocked
         Step 9. Process.start
         Step 10. ActivityThread.main
         Step 11. ActivityThread.attach
         Step 12. ActivityManagerService.attachApplication
   这五步是标准的Android应用程序启动步骤,具体可以参考
Android应用程序启动过程源代码分析 一文中的Step 23到Step 27,或者 Android系统在新进程中启动自定义服务过程(startService)的原理分析 一文中的Step 4到Step 9,这里就不再详细描述了。




本文转自 Luoshengyang 51CTO博客,原文链接:http://blog.51cto.com/shyluo/966987,如需转载请自行联系原作者
目录
相关文章
|
1月前
|
开发框架 前端开发 Android开发
Flutter 与原生模块(Android 和 iOS)之间的通信机制,包括方法调用、事件传递等,分析了通信的必要性、主要方式、数据传递、性能优化及错误处理,并通过实际案例展示了其应用效果,展望了未来的发展趋势
本文深入探讨了 Flutter 与原生模块(Android 和 iOS)之间的通信机制,包括方法调用、事件传递等,分析了通信的必要性、主要方式、数据传递、性能优化及错误处理,并通过实际案例展示了其应用效果,展望了未来的发展趋势。这对于实现高效的跨平台移动应用开发具有重要指导意义。
160 4
|
1月前
|
安全 Android开发 数据安全/隐私保护
深入探讨iOS与Android系统安全性对比分析
在移动操作系统领域,iOS和Android无疑是两大巨头。本文从技术角度出发,对这两个系统的架构、安全机制以及用户隐私保护等方面进行了详细的比较分析。通过深入探讨,我们旨在揭示两个系统在安全性方面的差异,并为用户提供一些实用的安全建议。
|
24天前
|
Java 开发工具 Android开发
安卓与iOS开发环境对比分析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自占据半壁江山。本文深入探讨了这两个平台的开发环境,从编程语言、开发工具到用户界面设计等多个角度进行比较。通过实际案例分析和代码示例,我们旨在为开发者提供一个清晰的指南,帮助他们根据项目需求和个人偏好做出明智的选择。无论你是初涉移动开发领域的新手,还是寻求跨平台解决方案的资深开发者,这篇文章都将为你提供宝贵的信息和启示。
29 8
|
28天前
|
安全 Android开发 数据安全/隐私保护
深入探索Android与iOS系统安全性的对比分析
在当今数字化时代,移动操作系统的安全已成为用户和开发者共同关注的重点。本文旨在通过比较Android与iOS两大主流操作系统在安全性方面的差异,揭示两者在设计理念、权限管理、应用审核机制等方面的不同之处。我们将探讨这些差异如何影响用户的安全体验以及可能带来的风险。
34 1
|
2月前
|
数据可视化 Android开发 开发者
安卓应用开发中的自定义View组件
【10月更文挑战第5天】在安卓应用开发中,自定义View组件是提升用户交互体验的利器。本篇将深入探讨如何从零开始创建自定义View,包括设计理念、实现步骤以及性能优化技巧,帮助开发者打造流畅且富有创意的用户界面。
101 0
|
2月前
|
XML 前端开发 Java
安卓应用开发中的自定义View组件
【10月更文挑战第5天】自定义View是安卓应用开发的一块基石,它为开发者提供了无限的可能。通过掌握其原理和实现方法,可以创造出既美观又实用的用户界面。本文将引导你了解自定义View的创建过程,包括绘制技巧、事件处理以及性能优化等关键步骤。
|
3月前
|
缓存 搜索推荐 Android开发
安卓应用开发中的自定义View组件实践
【9月更文挑战第10天】在安卓开发领域,自定义View是提升用户体验和实现界面个性化的重要手段。本文将通过一个实际案例,展示如何在安卓项目中创建和使用自定义View组件,包括设计思路、实现步骤以及可能遇到的问题和解决方案。文章不仅提供了代码示例,还深入探讨了自定义View的性能优化技巧,旨在帮助开发者更好地掌握这一技能。
|
4月前
|
XML 搜索推荐 Android开发
安卓开发中的自定义View组件实践
【8月更文挑战第30天】探索Android世界,自定义View是提升应用界面的关键。本文以简洁的语言带你了解如何创建自定义View,从基础到高级技巧,一步步打造个性化的UI组件。
|
5月前
|
机器学习/深度学习 人工智能 算法
探索AI在医疗影像分析中的应用探索安卓开发中的自定义View组件
【7月更文挑战第31天】随着人工智能技术的飞速发展,其在医疗健康领域的应用日益广泛。本文将聚焦于AI技术在医疗影像分析中的运用,探讨其如何通过深度学习模型提高诊断的准确性和效率。我们将介绍一些关键的深度学习算法,并通过实际代码示例展示这些算法是如何应用于医学影像的处理和分析中。文章旨在为读者提供对AI在医疗领域应用的深刻理解和实用知识。
50 0