Android通过共享用户ID来实现多Activity进程共享

本文探讨了Android中Activity在不同情况下的进程管理方式,包括同一Apk内Activity间的进程分配、不同Apk间Activity的进程隔离及共享机制,并讨论了如何通过配置实现进程的自定义。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

[b][size=large]1.同一Apk中的同一包中的Activity调用时进程状况验证[/size][/b]
[1]创建Project:
project name: [color=green]FirstProject[/color]
package:[color=red]com.demo[/color]
默认Activity:[color=green]MainActivity[/color]
[2]添加一个新的Activity:
name:[color=green]SecondActivity[/color]
[3]修改布局。在MainActivity布局中添加一个Button,当点击此Button时启动SecondActivity。在SecondActivity的布局中放置一个Textview,以证明SecondActivity已启动。
[4]运行程序,查看此Ap进程情况:
USER:[color=red]app_36[/color] PID:[color=darkblue]8360[/color] NAME:[color=green]com.demo[/color]
点击按钮,启动SecondActivity,再次查看进程情况:
USER:[color=red]app_36[/color] PID:[color=darkblue]8360[/color] NAME:[color=green]com.demo[/color]
[b][color=red]进程列表没有变化,两个Activity运行在同一进程中。[/color][/b]

[b][size=large]2.同一Apk中的不同包的Activity调用时进程状况验证[/size][/b]
[1]将SecondActivity挪到包com.demo.second中去,相应修改AndroidManifest.xml中的name为:com.demo.second.SecondActivity
[2]运行程序,查看此时进程情况:
USER:[color=red]app_36[/color] PID:[color=darkblue]10593[/color] NAME:[color=green]com.demo[/color]
[3]点击按钮启动SecondActivity,查看此时进程情况:
USER:[color=red]app_36[/color] PID:[color=darkblue]10593[/color] NAME:[color=green]com.demo[/color]
[b][color=red]进程列表没有变化,两个Activity运行在同一进程中。即进程name只受AndroidManifest.xml中manifset结点的package属性影响。[/color][/b]

[b][size=large]3.同一Apk中Activity process属性修改后进程状况验证[/size][/b]
[1]为SecondActivity添加process属性,其值为":abc",也可以随便是其他的":"开头的字符串,常见的是":remote":
<activity android:name="com.demo.second.SecondActivity" android:process=":abc"></activity>

[2]运行程序,查看进程情况:
USER:[color=red]app_36[/color] PID:[color=darkblue]12137[/color] NAME:[color=green]com.demo[/color]
[3]点击按钮,启动SecondActivity,查看进程情况:
USER:[color=red]app_36[/color] PID:[color=darkblue]12137[/color] NAME:[color=green]com.demo[/color]
USER:[color=red]app_36[/color] PID:[color=darkblue]12303[/color] NAME:[color=green]com.demo:abc[/color]
[b][color=red]进程表多了一项。两个Activity各自有一个进程,SecondActivity的进程名称为 包名+后缀[/color][/b]
[b][size=large]4.不同Apk中不同包名的Activity进程状况验证[/size][/b]
[1]运行FirstProject:
USER:[color=red]app_36[/color] PID:[color=darkblue]12137[/color] NAME:[color=green]com.demo[/color]
[2]创建SecondProject:
project name: [color=green]SecondProject[/color]
package:[color=red]com.demo2[/color]
默认Activity:[color=green]MainActivity[/color]
[3]运行SecondProject:
USER:[color=red]app_37[/color] PID:[color=darkblue]14191[/color] NAME:[color=green]com.demo2[/color]
[b][color=red]进程表多了一项。两个Activity各自有一个进程,同时其进程用户名、包名也不同,互不影响。[/color][/b]
[b][size=large]5.不同Apk,签名相同、包名相同的Activity进程状况验证[/size][/b]
[1]修改SecondProject的包也为com.demo,相应要修改AndroidManifest.xml内容。
[2]运行SecondProject,查看进程情况:
USER:[color=red]app_36[/color] PID:[color=darkblue]14944[/color] NAME:[color=green]com.demo[/color]
[b][color=red]进程表只有一项,但是实际上FirstProject此时已经被覆盖了,系统中只存在SecondProject了,因为模拟器调试时apk使用的签名key都是一样的,系统看到key一样,包名一样认为这个包就是FirstProject所以覆盖掉了。[/color][/b]
[b]可以通过DDMS复制/data/system/packages.xml查看一下内容。[/b]
<package name="com.demo" codePath="/data/app/com.demo.apk" system="false" ts="1279955425000" version="1" userId="10036">

这个文件里面,package name都是唯一的,同时可以看到用户名是通过userId来决定的。
[b][size=large]6.不同Apk,签名不相同,包名相同的Activity进程状况验证[/size][/b]
[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,提示安装失败。
[b][color=red]默认的Apk其安装时会分配新的UserId,即此时FirstProject以及SecondProject的UserId可以认为是不同的。
包名不同,则签名key是否相同无所谓,两个apk都可以安装。【第4个实验】
包名相同时,签名key相同则会覆盖【第5个实验】;签名不同则第二个apk安装会失败。【第6个实验】
[/color][/b]
[b][size=large]7.不同Apk,Share User Id相同,包名不同时进程情况分析[/size][/b]
[1]修改firstproject、secondproject的AndroidManifest.xml的manifset结点,增加属性
android:sharedUserId="com.demouser"

[2]修改secondproject的包为com.demo2,不然其会覆盖firsetproject。
[3]运行firsetproject、secondproject,查看进程列表:
USER:[color=red]app_35[/color] PID:[color=darkblue]19993[/color] NAME:[color=green]com.demo2[/color]
USER:[color=red]app_35[/color] PID:[color=darkblue]20045[/color] NAME:[color=green]com.demo2[/color]
仍然存在两个进程。进程的用户名一样,说明shareUserId确实有效了,但是进程名字一个不一样。
再次导出/data/system/packages.xml,查看其内容:
<package name="com.demo" codePath="/data/app/com.demo.apk" system="false" ts="1279957484000" version="1" sharedUserId="10035">

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

可以看到两个项目的UserId都是10035,确实是一样的。
[b][size=large]8.不同Apk,Share User Id相同,包名不同、指定Activity的process属性进程情况分析[/size][/b]
[1]修改SecondProject的MainActivity的process属性,指定绑定到进程名为com.demo的进程上:
<activity android:name=".MainActivity"
android:label="@string/app_name" android:process="com.demo">

[2]运行firstProject、SecondProject,查看进程情况:
USER:[color=red]app_35[/color] PID:[color=darkblue]21387[/color] NAME:[color=green]com.demo[/color]
两个Activity运行于同一个进程。
[b][size=large]9.不同Apk,Share User Id相同,包名不同、签名key不同[/size][/b]
经实验,安装第二个apk时会提示INSTALL_FAILED_UPDATE_INCOMPATIBLE错误,安装失败。

[b][color=green][size=large]
总结:
UserId不同时:
包名不同:未设定process属性时,各自的Activity在各自的进程。
即使process指定了包名,也不会和另一个用户的同名包共享进程。
包名相同:
签名相同:覆盖旧的同包名apk。
签名不同:新的apk会安装失败。【签名key一般都是不同的】
UserId相同时:
包名不同:未设定process属性时,各自的Activity在各自的进程。
process属性指定,则可以共享进程。
包名相同:
签名相同:覆盖旧的同包名apk。
签名不同:新的apk会安装失败。【签名key一般都是不同的】
[/size][/color][/b]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值