Android应用内多进程分析和研究
正常情况下,一个apk启动后只会运行在一个进程中,其进程名为AndroidManifest.xml文件中指定的应用包名,所有的基本组件都会在这个进程中运行。但是如果需要将某些组件(如Service、Activity等)运行在单独的进程中,就需要用到android:process属性了。我们可以为android的基础组件指定process属性来指定它们运行在指定进程中。
有什么好处
一般来说,Android应用多进程有三个好处。
1)我们知道Android系统对每个应用进程的内存占用是有限制的,而且占用内存越大的进程,通常被系统杀死的可能性越大。让一个组件运行在单独的进程中,可以减少主进程所占用的内存,降低被系统杀死的概率.
2)如果子进程因为某种原因崩溃了,不会直接导致主程序的崩溃,可以降低我们程序的崩溃率。
3)即使主进程退出了,我们的子进程仍然可以继续工作,假设子进程是推送服务,在主进程退出的情况下,仍然能够保证用户可以收到推送消息。
怎么来实现
对process属性的设置有两种形式:
第一种形式如 android:process=":remote",以冒号开头,冒号后面的字符串原则上是可以随意指定的。如果我们的包名为“com.example.processtest”,则实际的进程名为“com.example.processtest:remote”。这种设置形式表示该进程为当前应用的私有进程,其他应用的组件不可以和它跑在同一个进程中。
第二种情况如 android:process="com.example.processtest.remote",以小写字母开头,表示运行在一个以这个名字命名的全局进程中,其他应用通过设置相同的ShareUID可以和它跑在同一个进程。
下面通过一个例子来进行一下验证。我们定义两个类:ProcessTestActivity和ProcessTestService,然后在AndroidManifest.xml文件中增加这两个类,并为我们的Service指定一个process属性,代码如下:
- <?xml version="1.0" encoding="utf-8"?>
- <manifest xmlns:android="http://schemas.android.com/apk/res/android"
- package="com.example.processtest"
- android:versionCode="1"
- android:versionName="1.0" >
- <uses-sdk
- android:minSdkVersion="8"
- android:targetSdkVersion="19" />
- <application
- android:name="com.example.processtest.MyApplication"
- android:icon="@drawable/ic_launcher"
- android:label="@string/app_name">
- <activity
- android:name=".ProcessTestActivity"
- android:label="@string/app_name" >
- <intent-filter>
- <action android:name="android.intent.action.MAIN" />
- <category android:name="android.intent.category.LAUNCHER" />
- </intent-filter>
- </activity>
- <service
- android:name=".ProcessTestService"
- android:process=":remote">
- </service>
- </application>
- </manifest>
运行代码,通过DDMS进行观察,
我们可以看到两个进程,名字分别是“com.example.processtest”和“com.example.processtest:remote”,进程ID分别为2722和2739。
有哪些陷阱
我们已经开启了应用内多进程,那么,开启多进程是不是只是我们看到的这么简单呢?其实这里面会有一些陷阱,稍微不注意就会陷入其中。我们首先要明确的一点是进程间的内存空间时不可见的。从而,开启多进程后,我们需要面临这样几个问题:
1)Application的多次重建。
2)静态成员的失效。
3)文件共享问题。
4)断点调试问题。
我们先通过一个简单的例子来看一下第一种情况。
Manifest文件如上面提到的,定义了两个类:ProcessTestActivity和ProcessTestService,我们只是在Activity的onCreate方法中直接启动了该Service,同时,我们自定义了自己的Application类。代码如下:
- public class MyApplication extends Application {
- public static final String TAG = "viclee";
- @Override
- public void onCreate() {
- super.onCreate();
- int pid = android.os.Process.myPid();
- Log.d(TAG, "MyApplication onCreate");
- Log.d(TAG, "MyApplication pid is " + pid);
- }
- }
- public class ProcessTestActivity extends Activity {
- public final static String TAG = "viclee";
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_process_test);
- Log.i(TAG, "ProcessTestActivity onCreate");
- this.startService(new Intent(this, ProcessTestService.class));
- }
- }
- public class ProcessTestService extends Service {
- public static final String TAG = "viclee";
- @Override
- public void onCreate() {
- Log.i(TAG, "ProcessTestService onCreate");
- }
- @Override
- public IBinder onBind(Intent arg0) {
- return null;
- }
- }
执行上面这段代码,查看打印信息:
我们发现MyApplication的onCreate方法调用了两次,分别是在启动ProcessTestActivity和ProcessTestService的时候,而且我们发现打印出来的pid也不相同。由于通常会在Application的onCreate方法中做一些全局的初始化操作,它被初始化多次是完全没有必要的。出现这种情况,是由于即使是通过指定process属性启动新进程的情况下,系统也会新建一个独立的虚拟机,自然需要重新初始化一遍Application。那么怎么来解决这个问题呢?
我们可以通过在自定义的Application中通过进程名来区分当前是哪个进程,然后单独进行相应的逻辑处理。
- public class MyApplication extends Application {
- public static final String TAG = "viclee";
- @Override
- public void onCreate() {
- super.onCreate();
- int pid = android.os.Process.myPid();
- Log.d(TAG, "MyApplication onCreate");
- Log.d(TAG, "MyApplication pid is " + pid);
- ActivityManager am = (ActivityManager) getSystemService(Context.ACTIVITY_SERVICE);
- List<ActivityManager.RunningAppProcessInfo> runningApps = am.getRunningAppProcesses();
- if (runningApps != null && !runningApps.isEmpty()) {
- for (ActivityManager.RunningAppProcessInfo procInfo : runningApps) {
- if (procInfo.pid == pid) {
- if (procInfo.processName.equals("com.example.processtest")) {
- Log.d(TAG, "process name is " + procInfo.processName);
- } else if (procInfo.processName.equals("com.example.processtest:remote")) {
- Log.d(TAG, "process name is " + procInfo.processName);
- }
- }
- }
- }
- }
- }
运行之后,查看Log信息,
图中可以看出,不同的进程执行了不同的代码逻辑,可以通过这种方式来区分不同的进程需要完成的初始化工作。
下面我们来看第二个问题,将之前定义的Activity和Service的代码进行简单的修改,代码如下:
- public class ProcessTestActivity extends Activity {
- public final static String TAG = "viclee";
- public static boolean processFlag = false;
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_process_test);
- processFlag = true;
- Log.i(TAG, "ProcessTestActivity onCreate");
- this.startService(new Intent(this, ProcessTestService.class));
- }
- }
- public class ProcessTestService extends Service {
- public static final String TAG = "viclee";
- @Override
- public void onCreate() {
- Log.i(TAG, "ProcessTestService onCreate");
- Log.i(TAG, "ProcessTestActivity.processFlag is " + ProcessTestActivity.processFlag);
- }
- @Override
- public IBinder onBind(Intent arg0) {
- return null;
- }
- }
重新执行代码,打印Log
从上面的代码和执行结果看,我们在Activity中定义了一个标志processFlag并在onCreate中修改了它的值为true,然后启动Service,但是在Service中读到这个值却为false。按照正常的逻辑,静态变量是可以在应用的所有地方共享的,但是设置了process属性后,产生了两个隔离的内存空间,一个内存空间里值的修改并不会影响到另外一个内存空间。
第三个问题是文件共享问题。多进程情况下会出现两个进程在同一时刻访问同一个数据库文件的情况。这就可能造成资源的竞争访问,导致诸如数据库损坏、数据丢失等。在多线程的情况下我们有锁机制控制资源的共享,但是在多进程中比较难,虽然有文件锁、排队等机制,但是在Android里很难实现。解决办法就是多进程的时候不并发访问同一个文件,比如子进程涉及到操作数据库,就可以考虑调用主进程进行数据库的操作。
最后是断点调试的问题。调试就是跟踪程序运行过程中的堆栈信息,由于每个进程都有自己独立的内存空间和各自的堆栈,无法实现在不同的进程间调试。不过可以通过下面的方式实现:调试时去掉AndroidManifest.xml中android:process标签,这样保证调试状态下是在同一进程中,堆栈信息是连贯的。待调试完成后,再将标签复原。
总结
从上面的例子中我们可以看到,android实现应用内多进程并不是简单的设置属性process就可以了,而是会产生很多特殊的问题。像前面提到的,android启动多进程模式后,不仅静态变量会失效,而且类似的如同步锁机制、单例模式也会存在同样的问题。这就需要我们在使用的时候多加注意。而且设置多进程之后,各个进程间就无法直接相互访问数据,只能通过AIDL等进程间通信方式来交换数据。
Android应用内多进程分析和研究相关推荐
- Android 应用内多进程实现 单APK应用多进程
Android平台支持多进程通信,也支持应用内实现多进程,Android中,默认一个APK包就对应一个进程,其进程名就为AndroidManifest.xml文件中 指定的package名.我们可以通 ...
- 全球与中国厂内物流自动化市场深度研究分析报告
[报告篇幅]:108 [报告图表数]:144 [报告出版时间]:2021年12月 报告摘要 本文研究全球及中国市场厂内物流自动化现状及未来发展趋势,侧重分析全球及中国市场的主要企业,同时对比北美.欧洲 ...
- Android逆向与病毒分析
本文由同程旅游安全团队对内移动安全培训的PPT整理而来,面向对象为对移动安全感兴趣的研发同事,所以讲的有些宽泛.介绍了入门Android逆向需要掌握的一些知识点, 通过简单的几个案例讲解Android ...
- Android 应用性能优化(4)---Android App性能评测分析-启动时间篇
Android App性能评测分析-启动时间篇 1.前言 随着项目版本的迭代,App的性能问题会逐渐暴露出来,而好的用户体验与性能表现紧密相关,性能问题从应用的启动优化开始,下面会根据实际app性能测 ...
- 如何分析和研究Log文件 ,如何看日志信息
如何分析和研究Log文件 ,如何看日志信息 . Log 在android中的地位非常重要,要是作为一个android程序员不能过分析log这关,算是android没有入门吧 . 下面我们就来说说如何处 ...
- 可能是最详细的Android图片压缩原理分析(一)—— Android图片压缩必备基础知识
本篇文章已授权微信公众号guolin_blog(郭霖)独家发布 稀土掘金链接 前言: 最近在研究图片压缩原理,看了大量资料,从上层尺寸压缩.质量压缩原理到下层的哈夫曼压缩,走成华大道,然后去二仙桥,全 ...
- Android service启动流程分析.
文章仅仅用于个人的学习记录,基本上内容都是网上各个大神的杰作,此处摘录过来以自己的理解学习方式记录一下. 参考链接: https://my.oschina.net/youranhongcha/blog ...
- Android WiFi —softAP流程分析
Android WiFi - Ap功能实现与源码分析 0. 前言 wifiAp的ip WifiAp的config分析 2.1 默认的config 2.2 修改wifiAp的config配置流程 开启/ ...
- Android Hal层简要分析
Android Hal层简要分析 Android Hal层(即 Hardware Abstraction Layer)是Google开发的Android系统里上层应用对底层硬件操作屏蔽的一个软件层次, ...
最新文章
- 今天看论坛,有这样一句话,深有同感,还是家里好
- 谷歌大脑最新研究:用AutoML的方式自动学习Dropout模式,再也不用手动设计
- oracle rac 组播,Oracle 11g RAC安装测试组播
- IOS开发网络第一天之06线程之间的通信
- Javascript编写的简易计算器
- web基础html元素制作web
- opencv 画矩形_图像处理之OpenCV的基础使用补充
- 日常工作用Python能解决哪些问题?
- surfaceview结束后怎么处理_污泥压滤机处理后的污泥怎么处置
- 基于TCP原理,采用Socket通信技术,实现聊天室
- (王道408考研操作系统)第四章文件管理-第一节7:文件共享
- HDU5470 Typewriter SAM 动态规划 单调队列
- TCP系列05—连接管理—4、TCP连接的ISN、连接建立超时及TCP的长短连接
- 实现简单render函数
- 从键盘输入10个整数,求其平均值
- 一、 Vue.js简介
- JasperReport生成pdf文件 Java开发pdf文件 pdf文件生成及下载
- 机器人的自述作文_关于机器人的说明作文
- 企业微信——定时群机器人布置
- HDU 新生赛 油菜花王国(并查集)
热门文章
- 无重复最长子串python_leetCode 无重复字符的最长子串 python(Longest Substring Without Repeating Characters) python...
- 全球科技大会鸿蒙,华为开发者大会:全球瞩目的“鸿蒙”面子和里子大格局
- eureka多台注册中心_微服务实战系列(五)-注册中心Eureka与nacos区别
- 计算机二级c语言复习计划,全国计算机等级考试二级C语言考试复习资料与复习计划以与考试要点.doc...
- 进程、线程、多线程、并发、并行学习记录
- ROS中RVIS导入机器人模型,添加摄像头,雷达,Kinect
- sqlserver 无法远程连接到服务器,SQLServer2019无法连接远程服务器
- win7 计算机右键没有属性,系统之家windows7旗舰版计算机上右键菜单没有属性怎么办?...
- MobileNets论文阅读笔记
- 关于arm处理器 内存编址模式 与 字节对齐方式 (转)