Android中多進(jìn)程是指一個(gè)應(yīng)用存在多個(gè)進(jìn)程的情況。
為何使用多進(jìn)程隙赁?
多進(jìn)程的情況分為兩種杖狼;
第一種情況是一個(gè)應(yīng)用因?yàn)槟承┰蜃陨硇枰捎枚噙M(jìn)程模式實(shí)現(xiàn),比如有些模塊需要在單獨(dú)的進(jìn)程的運(yùn)行猿挚,又或者為了加大一個(gè)應(yīng)用可使用的內(nèi)存通過(guò)多進(jìn)程獲取更大的運(yùn)行內(nèi)存空間等咐旧。Android對(duì)單個(gè)應(yīng)用使用的內(nèi)存進(jìn)行了限制,不同設(shè)備不同大小绩蜻。
第二種情況:當(dāng)前應(yīng)用需要向其他應(yīng)用獲取數(shù)據(jù)铣墨,由于兩個(gè)應(yīng)用,需要跨進(jìn)程的方式來(lái)獲取數(shù)據(jù)等办绝。
開啟多進(jìn)程的模式
在AndroidMenifest中指定Android:process屬性伊约。
<activity
android:name=".SecondActivity"
android:process=":twoprocess"></activity>
<activity
android:name=".ThirdActivity"
android:process="comzzj.myappartdevelopmentdemo.thirdprocess"></activity>
看一下運(yùn)行起來(lái)進(jìn)程的名字:
默認(rèn)進(jìn)程:
secondActivity的進(jìn)程:
thirdActivity的進(jìn)程:
從進(jìn)程的名字可以看出以“:”開頭的進(jìn)程屬于應(yīng)用的私有進(jìn)程,其他應(yīng)用的組件不可以和它跑在同一個(gè)進(jìn)程中孕蝉,而進(jìn)程名不以“:”開頭的進(jìn)程屬于全局進(jìn)程碱妆,其他應(yīng)用通過(guò)shareUID方式可以和它跑在同一個(gè)進(jìn)程中。Android系統(tǒng)會(huì)為每個(gè)應(yīng)用分配一個(gè)唯一的UID昔驱,具有唯一UID的應(yīng)用才可以共享數(shù)據(jù)疹尾。
多進(jìn)程模式下的運(yùn)行機(jī)制
下面通過(guò)簡(jiǎn)單的例子看一下通過(guò)靜態(tài)成員變量的值,查看多線程下運(yùn)行的機(jī)制
public class ConstantManager {
public static int CONSTANT_ID = 1;
}
mainActivity的onCreate方法
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ConstantManager.CONSTANT_ID = 2;
Log.d("CONSTANT_ID","MainActivity.CONSTANT_ID:"+ConstantManager.CONSTANT_ID);
}
SecondActivity的onCreate方法
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.act_second);
Log.d("CONSTANT_ID","SecondActivity.CONSTANT_ID:"+ConstantManager.CONSTANT_ID);
}
從打印日志可以看出骤肛,在默認(rèn)進(jìn)程中修改的值只會(huì)影響當(dāng)前的進(jìn)程纳本,對(duì)其他進(jìn)程不會(huì)造成任何影響。
Android為每一個(gè)應(yīng)用分配了獨(dú)立的虛擬機(jī)腋颠,或者說(shuō)為每一個(gè)進(jìn)程分配了一個(gè)獨(dú)立的虛擬機(jī)繁成,不同虛擬機(jī)在內(nèi)存分配上有不同的地址空間,這就導(dǎo)致在不同虛擬機(jī)中訪問(wèn)同一個(gè)類對(duì)象會(huì)產(chǎn)生多份副本淑玫。
一般來(lái)說(shuō)巾腕,使用多進(jìn)程會(huì)造成如下幾個(gè)方面的問(wèn)題:
1、靜態(tài)成員和單例模式無(wú)效
2絮蒿、線程同步機(jī)制完全失效
3尊搬、SharedPreferences的可靠性下降
4、Application的多次創(chuàng)建土涝。
前三個(gè)問(wèn)題很容易理解佛寿,多進(jìn)程下去同時(shí)操作數(shù)據(jù)和類,肯定會(huì)產(chǎn)生以上問(wèn)題但壮。
我們主要通過(guò)簡(jiǎn)單測(cè)試?yán)斫庀碌谒膫€(gè)問(wèn)題怎么多次創(chuàng)建呢冀泻?
獲取當(dāng)前運(yùn)行進(jìn)程的名稱
public static String getProcessName(Context cxt, int pid) {
ActivityManager am = (ActivityManager) cxt.getSystemService(Context.ACTIVITY_SERVICE);
List<RunningAppProcessInfo> runningApps = am.getRunningAppProcesses();
if (runningApps == null) {
return null;
}
for (RunningAppProcessInfo procInfo : runningApps) {
if (procInfo.pid == pid) {
return procInfo.processName;
}
}
return null;
}
public class MyApplication extends Application {
@Override
public void onCreate() {
super.onCreate();
Log.d("MyApplication","MyApplication.process:"+getProcessName(getApplicationContext(), Process.myPid()));
}
從log中可以看出常侣,Application執(zhí)行了三次onCreate,并且每次的進(jìn)程名稱和進(jìn)程id都不一樣弹渔。
雖然我們分析了這么多多進(jìn)程帶來(lái)的問(wèn)題胳施,但是不能因?yàn)槎噙M(jìn)程有很多問(wèn)題就不去正視它,為了解決這個(gè)問(wèn)題肢专,系統(tǒng)為我們提供了很多跨進(jìn)程通信的方法巾乳。雖然不能直接共享內(nèi)存,但是可以通過(guò)跨進(jìn)程通信(IPC)實(shí)現(xiàn)數(shù)據(jù)交互鸟召。