apk一般占一个dalvik,一个进程,一个task。通过设置也可以多个进程,占多个task。

task是一个activity的栈,其中"可能"含有来自多个App的activity

默认情况下,同一个应用程序中的所有组件运行在同一个进程中,而且绝大多数的应用程序也都是这样的。但是,如果我们想要控制让某个特定的组件属于某个进程,我们可以在manifest文件中进行配置。 
在 每种组件元素(activity、service、receiver、provider)的manifest条目中,都支持一个 “android:process”的属性,通过这个属性,我们可以指定某个组件运行的进程。我们可以通过设置这个属性,让每个组件运行在它自己的进程 中,也可以只让某些组件共享一个进程。我们要可以通过设置“android:process”属性,让不同应用程序中的组件运行在相同的进程中,这些应用 程序共享相同的Linux用户ID,拥有相同的证书。 
<application>元素也有一个“android:process”属性,可以设置一个应用于全部组件的默认值。 
    当可用内存数量低,而一些与用户即时交互的进程又需要内存时,Android随时可能会终止某个进程。运行在被终止的进程中的组件会因此被销毁,但是,当再次需要这些组件工作时,就会再启动一个进程。 
    在决定要终止哪个进程时,Android系统会权衡它们对于用户的重要性。例如,相较于运行可见activities的进程,终止一个运行不可见activities的进程会更加合理。是否终止一个进程,依赖于运行在这个进程中的组件的状态。 
如果不能将两个activity放入同一个application中的话,可以通过在各自的manifest中设置以下属性,让这两个activity强制运行在同一个进程中,从而可以充分利用进程内共享的资源,减少内存占用:

  1. (1)设置相同的User Id:
  2. <manifest android:sharedUserId="aaa.bbb"
  3. (2)被调用的activity设置以下属性:
  4. <activity android:multiprocess="true"
  5. 或者
  6. <activity android:process="com.cienet.test"

对于3D OpenGL程序,修改以上属性后,被调用的activity的内存占用会明显减少,比如:30MB -> 2MB。

可以做如下测试如何共享一个进程

  1. 加入android:sharedUserId="aaa.bbb"没有
  2. android:process="com.cienet.test"
  3. 07-12 15:42:34.781: ERROR/Dean(7615): Binder.getCallingPid()7615
  4. 07-12 15:42:34.785: ERROR/Dean(7615): runningappprocessinfo.pid:7615runningappprocessinfo.processNamecom.cienet.testa
  5. 07-12 15:42:34.785: ERROR/Dean(7615): pkgname:7615:com.cienet.testa
  6. 07-12 15:42:24.722: ERROR/Dean(7602): Binder.getCallingPid()7602
  7. 07-12 15:42:24.726: ERROR/Dean(7602): runningappprocessinfo.pid:7602runningappprocessinfo.processNamecom.cienet.testb
  8. 07-12 15:42:24.726: ERROR/Dean(7602): pkgname:7602:com.cienet.testb
  9. 加入android:process="com.cienet.test"到B
  10. 07-12 15:46:41.933: ERROR/Dean(7738): Binder.getCallingPid()7738
  11. 07-12 15:46:41.937: ERROR/Dean(7738): runningappprocessinfo.pid:7738runningappprocessinfo.processNamecom.cienet.test
  12. 07-12 15:46:41.937: ERROR/Dean(7738): pkgname:7738:com.cienet.testb
  13. 07-12 15:47:13.246: ERROR/Dean(7761): Binder.getCallingPid()7761
  14. 07-12 15:47:13.246: ERROR/Dean(7761): runningappprocessinfo.pid:7761runningappprocessinfo.processNamecom.cienet.testa
  15. 07-12 15:47:13.250: ERROR/Dean(7761): pkgname:7761:com.cienet.testa
  16. 同时加入android:process="com.cienet.test"到A
  17. 07-12 15:50:24.988: ERROR/Dean(7878): getApplicationInfo:publicSourceDir/data/app/com.cienet.testa-1.apk
  18. 07-12 15:50:24.988: ERROR/Dean(7878): getApplicationInfo:descriptionRes0
  19. 07-12 15:50:24.988: ERROR/Dean(7878): Binder.getCallingPid()7878
  20. 07-12 15:50:24.992: ERROR/Dean(7878): runningappprocessinfo.pid:7878runningappprocessinfo.processNamecom.cienet.test
  21. 07-12 15:50:24.992: ERROR/Dean(7878): pkgname:7878:com.cienet.testa
  22. 07-12 15:50:46.964: ERROR/Dean(7878): getApplicationInfo:publicSourceDir/data/app/com.cienet.testb-1.apk
  23. 07-12 15:50:46.964: ERROR/Dean(7878): getApplicationInfo:descriptionRes0
  24. 07-12 15:50:46.964: ERROR/Dean(7878): Binder.getCallingPid()7878
  25. 07-12 15:50:46.968: ERROR/Dean(7878): runningappprocessinfo.pid:7878runningappprocessinfo.processNamecom.cienet.test
  26. 07-12 15:50:46.972: ERROR/Dean(7878): pkgname:7878:com.cienet.testa
  27. 去掉android:sharedUserId="aaa.bbb"
  28. 07-12 15:52:29.574: ERROR/Dean(7967): getApplicationInfo:publicSourceDir/data/app/com.cienet.testb-2.apk
  29. 07-12 15:52:29.578: ERROR/Dean(7967): getApplicationInfo:descriptionRes0
  30. 07-12 15:52:29.578: ERROR/Dean(7967): Binder.getCallingPid()7967
  31. 07-12 15:52:29.578: ERROR/Dean(7967): runningappprocessinfo.pid:7967runningappprocessinfo.processNamecom.cienet.test
  32. 07-12 15:52:29.578: ERROR/Dean(7967): pkgname:7967:com.cienet.testb
  33. 07-12 15:53:06.730: ERROR/Dean(7990): getApplicationInfo:publicSourceDir/data/app/com.cienet.testa-2.apk
  34. 07-12 15:53:06.730: ERROR/Dean(7990): getApplicationInfo:descriptionRes0
  35. 07-12 15:53:06.730: ERROR/Dean(7990): Binder.getCallingPid()7990
  36. 07-12 15:53:06.734: ERROR/Dean(7990): runningappprocessinfo.pid:7990runningappprocessinfo.processNamecom.cienet.test
  37. 07-12 15:53:06.734: ERROR/Dean(7990): pkgname:7990:com.cienet.testa

如上证明那个APK共享进程要两个条件(1)设置相同的User Id:

(2)被调用的activity设置以下属性:

  1. <activity android:multiprocess="true"
  2. 或者
  3. <activity android:process="com.cienet.test"

1. 同一Apk中的同一包中的多个Activity调用时进程状况验证

[1]创建Project:

project name: FirstProject

package     : com.demo

默认Activity : MainActivity

[2]添加一个新的Activity:

name: SecondActivity

[3]修改布局。在MainActivity布局中添加一个Button,当点击此Button时启动SecondActivity。在SecondActivity的布局中放置一个Textview,以证明SecondActivity已启动。

[4]运行程序,查看此App进程情况:

USER:app_36  PID:8360  NAME:com.demo

[5]点击按钮,启动SecondActivity,再次查看进程情况:

USER:app_36  PID:8360  NAME:com.demo

结论:进程列表没有变化,两个Activity运行在同一进程中。

2. 同一Apk中的不同包的Activity调用时进程状况验证

[1]将SecondActivity挪到包com.demo.second中去,相应修改AndroidManifest.xml中的name为:com.demo.second.SecondActivity

[2]运行程序,查看此时进程情况:

USER:app_36  PID:10593  NAME:com.demo

[3]点击按钮启动SecondActivity,查看此时进程情况:

USER:app_36  PID:10593  NAME:com.demo

结论:进程列表没有变化,两个Activity运行在同一进程中。即进程name只受AndroidManifest.xml中manifset结点的package属性影响。

3. 同一Apk中Activity process属性修改后进程状况验证

[1]为SecondActivity添加process属性,其值为":abc",也可以随便是其他的":"开头的字符串,常见的名字是":remote":

<activity android:name="com.demo.second.SecondActivity" android:process=":abc"> </activity>

[2]运行程序,查看进程情况:

USER:app_36  PID:12137  NAME:com.demo

[3]点击按钮,启动SecondActivity,查看进程情况:

USER:app_36  PID:12137  NAME:com.demo

USER:app_36  PID:12303  NAME:com.demo:abc

结论:进程表多了一项。两个Activity各自有一个进程,SecondActivity的进程名称为 包名+后缀。

4. 不同Apk中不同包名的Activity进程状况验证 

[1]运行FirstProject:

USER:app_36  PID:12137  NAME:com.demo

[2]创建SecondProject:

project name: SecondProject

package:com.demo2

默认Activity:MainActivity

[3]运行SecondProject:

USER:app_37  PID:14191  NAME:com.demo2

结论:进程表多了一项。两个Activity各自有一个进程,同时其进程用户id、包名也不同,互不影响。

5. 不同Apk,签名相同、包名相同的Activity进程状况验证 

[1]修改SecondProject的包也为com.demo,相应要修改AndroidManifest.xml内容。

[2]运行SecondProject,查看进程情况:

USER:app_36  PID:14944  NAME:com.demo

结论:进程表只有一项,但是实际上FirstProject此时已经被覆盖了,系统中只存在SecondProject了,因为模拟器调试时apk使用的签名key都是一样的,系统看到key一样,包名一样认为这个包就是FirstProject所以覆盖掉了。

可以通过DDMS复制/data/system/packages.xml查看一下内容:

<package name="com.demo" codePath="/data/app/com.demo.apk" system="false" ts="1279955425000" version="1" userId="10036">

这个文件里面,package name都是唯一的,同时可以看到用户名是通过userId来决定的。

6.不同Apk,签名不相同,包名相同的Activity进程状况验证

[1]在Eclipse的Package Explorer导航树中选中FirstProject,点右键。

[2]Android tools-->Export Signed Application Package,按照向导创建一个用指定key签名的apk包。

[3]同样导出Second Project。

[4]切换窗口到模拟器,按Home键-->按Menu键-->设置-->应用程序-->管理应用程序-->SecondProject-->卸载。这是为了用命令行安装做准备。

[5]启动一个命令行窗口,执行adb install firstproject.apk,会提示成功安装。

[6]执行adb install secondproject.apk,提示安装失败。

结论:

1> 默认的Apk其安装时会分配新的UserId,即此时FirstProject以及SecondProject的UserId可以认为是不同的。

2> 包名不同,则签名key是否相同无所谓,两个apk都可以安装。【第4个实验】

3> 包名相同时,签名key相同则会覆盖【第5个实验】;签名不同则第二个apk安装会失败。【第6个实验】

7.不同Apk,Share User Id相同,包名不同时进程情况分析

[1]修改firstproject、secondproject的AndroidManifest.xml的manifset结点,增加属性

android:sharedUserId="com.demouser"

[2]修改secondproject的包为com.demo2,不然其会覆盖firsetproject。

[3]运行firsetproject、secondproject,查看进程列表:

USER:app_35  PID:19993  NAME:com.demo2

USER:app_35  PID:20045  NAME:com.demo2

结论:

仍然存在两个进程。但是进程的用户名一样,说明shareUserId确实有效了,进程pid不相同。

再次导出/data/system/packages.xml,查看其内容,可以看到两个项目的UserId都是10035,确实是一样的:

<package name="com.demo" codePath="/data/app/com.demo.apk" system="false" ts="1279957484000" version="1" sharedUserId="10035"> <packagename="com.demo2" codePath="/data/app/com.demo2.apk" system="false" ts="1279957473000" version="1" sharedUserId="10035">

8.不同Apk,Share User Id相同,包名不同、指定Activity的process属性进程情况分析

[1]修改SecondProject的MainActivity的process属性,指定绑定到进程名为com.demo的进程上:

<activity android:name=".MainActivity" android:label="@string/app_name" android:process="com.demo">

[2]运行firstProject、SecondProject,查看进程情况:

USER:app_35  PID:21387  NAME:com.demo

结论:两个Activity运行于同一个进程。

9.不同Apk,Share User Id相同,包名不同、签名key不同

经实验,安装第二个apk时会提示INSTALL_FAILED_UPDATE_INCOMPATIBLE错误,安装失败。

总结:

UserId不同时: 

    包名不同:

        未设定process属性时,各自的Activity在各自的进程。即使process指定了包名,也不会和另一个用户的同名包共享进程。 

    包名相同: 

        签名相同:覆盖旧的同包名apk。签名不同:新的apk会安装失败。【签名key一般都是不同的】

UserId相同时: 

    包名不同:

        未设定process属性时,各自的Activity在各自的进程。process属性指定,则可以共享进程。 

    包名相同: 

        签名相同:覆盖旧的同包名apk。签名不同:新的apk会安装失败。【签名key一般都是不同的】

 

 
 

apk,task,android:process与android:sharedUserId的区别的更多相关文章

  1. android:process结合activity启动模式的一次实践

    会有这样的场景,一个应用崩溃了,而导致的该应用崩溃的原因是,该应用占用的内存大小超过了系统分配给它的最大堆大小.对象的分配,是发生在堆(heap)上面的,系统分配给每个应用的最大堆大小是固定的. 假设 ...

  2. android.process.acore和system进程

    从源码看来,android.process.acore进程应该是一些基本功能的载入程序. android-4.3_r2.2中,它包括以下项目: 1.UserDictionaryProvider < ...

  3. Android中的APK,TASK,PROCESS,USERID之间的关系

    开发Android已经有一段时间了,今天接触到底层的东西,所以对于进程,用户的id以及Android中的Task,Apk之间的关系,要做一个研究,下面就是研究结果: apk一般占一个dalvik,一个 ...

  4. Error:Execution failed for task ':app:dexDebug'. > com.android.ide.common.process.ProcessException

    异常Log: Error:Execution failed for task ‘:app:dexDebug’. > com.android.ide.common.process.ProcessE ...

  5. Error:Execution failed for task ':app:transformClassesWithDexForDebug'. > com.android.build.api.transform.TransformException: com.android.ide.common.process.ProcessException: java.util.concurrent.Exec

    Error:Execution failed for task ':app:transformClassesWithDexForDebug'.> com.android.build.api.tr ...

  6. Error:Execution failed for task ':app:dexDebug'. > com.android.ide.common.process.ProcessException总结

    最新项目中遇到了 Error:Execution failed for task ':app:dexDebug'. > com.android.ide.common.process.Proces ...

  7. No permission to write APN settings: Neither user *** nor current process has android.permission.WRITE_APN_SETTINGS.

    在ICS40以前的版本中,如果程序需要设置APN,只需要在AndroidManifest文件中声明这个权限即可.在40的机器上运行则会抛出以下异常:java.lang.SecurityExceptio ...

  8. android:process为耗资源操作指定一个新进程

    当有一些耗费内存比较多的操作时,可以通过android:process指定一个新的进程.保证程序运行. 例如: 一个后台长期运行的service: <service android:name=& ...

  9. 【转】Android多进程总结一:生成多进程(android:process属性)

    前言 正常情况下,一个apk启动后只会运行在一个进程中,其进程名为apk的包名,所有的组件都会在这个进程中运行,以下为DDMS的进程截屏: com.biyou.multiprocess为进程名,也是a ...

随机推荐

  1. Bash中的$符号

    脚本名称:$0 PID:$$ 参数个数:$# 脚本返回值:$? 第x个参数:$x 第10个以上的参数加大括号:${10} 所有参数:$@ #!/bin/bash echo "The prog ...

  2. JS实例(一)

    一:单选按钮,选择同意,提交变为可用,反正提交不可用: HTML里面代码: <form id="f1" name="f1"> <input t ...

  3. PHP之APC缓存详细介绍

    1.APC缓存简介 APC,全称是Alternative PHP Cache,官方翻译叫"可选PHP缓存".它为我们提供了缓存和优化PHP的中间代码的框架. APC的缓存分两部分: ...

  4. C#语法糖之第四篇: 扩展方法

    今天继续分享C#4.0语法糖的扩展方法,这个方法也是我本人比较喜欢的方法.大家先想想比如我们以前写的原始类型不能满足现在的需求,而需要在该类型中添加新的方法来实现时大家会怎么做.我先说一下我没有学习到 ...

  5. SqlSugar-事务操作

    一.事务操作实例 特别说明: 1.特别说明:在事务中,默认情况下是使用锁的,也就是说在当前事务没有结束前,其他的任何查询都需要等待 2.ReadCommitted:在正在读取数据时保持共享锁,以避免脏 ...

  6. 如何快速的将Centos6.7快速升级3.10.9

    参考文档:http://www.xiexianbin.cn/linux/2015/10/15/quickly-upgrade-centos6.5-kernel-from-2.6.32-to-3.10. ...

  7. Access restriction:The type JPEGCodec is not accessible due to restriction on required library C:\Program Files\Java\jre6\lib\rt.jar

    解决方法: Project -> Properties -> libraries, 先remove掉JRE System Library,然后再Add Library重新加入. ===== ...

  8. MySQL 时间戳(Timestamp)函数

    1. MySQL 获得当前时间戳函数:current_timestamp, current_timestamp() mysql> select current_timestamp, curren ...

  9. localStorage.ie6.js

    !window.localStorage && function() { window.localStorage = {}; var prefix = 'data-userdata' ...

  10. Photon开发实战(2)——开发框架、第一个Photon程序

    Photon基础开发框架 Photon (v4)的基本框架.开发框架主要Photon和游戏逻辑(C#)两个部分,如下图最新的Photon v4支持的4种底层协议,游戏开发逻辑Photon目前主要划分为 ...