Android中應(yīng)用多進(jìn)程的整理總結(jié)
前言
在計(jì)算機(jī)操作系統(tǒng)中,進(jìn)程是進(jìn)行資源分配和調(diào)度的基本單位。這對(duì)于基于Linux內(nèi)核的Android系統(tǒng)也不例外。在Android的設(shè)計(jì)中,一個(gè)應(yīng)用默認(rèn)有一個(gè)(主)進(jìn)程。但是我們通過(guò)配置可以實(shí)現(xiàn)一個(gè)應(yīng)用對(duì)應(yīng)多個(gè)進(jìn)程。
本文將試圖對(duì)于Android中應(yīng)用多進(jìn)程做一些整理總結(jié)。
android:process
- 應(yīng)用實(shí)現(xiàn)多進(jìn)程需要依賴于
android:process
這個(gè)屬性 - 適用元素:Application, Activity, BroadcastReceiver, Service, ContentProvider。
- 通常情況下,這個(gè)屬性的值應(yīng)該是”:“開(kāi)頭。表示這個(gè)進(jìn)程是應(yīng)用私有的。無(wú)法在在跨應(yīng)用之間共用。
- 如果該屬性值以小寫字母開(kāi)頭,表示這個(gè)進(jìn)程為全局進(jìn)程??梢员欢鄠€(gè)應(yīng)用共用。(文章結(jié)尾會(huì)探討這個(gè)問(wèn)題)
一個(gè)應(yīng)用 android:process
簡(jiǎn)單示例
<activity android:name=".MusicPlayerActivity" android:process=":music"/> <activity android:name=".AnotherActivity" android:process="droidyue.com"/>
應(yīng)用多進(jìn)程有什么好處
增加App可用內(nèi)存
在Android中,默認(rèn)情況下系統(tǒng)會(huì)為每個(gè)App分配一定大小的內(nèi)存。比如從最早的16M到后面的32M或者48M等。具體的內(nèi)存大小取決于硬件和系統(tǒng)版本。
這些有限的內(nèi)存對(duì)于普通的App還算是夠用,但是對(duì)于展示大量圖片的應(yīng)用來(lái)說(shuō),顯得實(shí)在是捉襟見(jiàn)肘。
仔細(xì)研究一下,你會(huì)發(fā)現(xiàn)原來(lái)系統(tǒng)的這個(gè)限制是作用于進(jìn)程的(畢竟進(jìn)程是作為資源分配的基本單位)。意思就是說(shuō),如果一個(gè)應(yīng)用實(shí)現(xiàn)多個(gè)進(jìn)程,那么這個(gè)應(yīng)用可以獲得更多的內(nèi)存。
于是,增加App可用內(nèi)存成了應(yīng)用多進(jìn)程的重要原因。
獨(dú)立于主進(jìn)程
除了增加App可用內(nèi)存之外,確保使用多進(jìn)程,可以獨(dú)立于主進(jìn)程,確保某些任務(wù)的執(zhí)行和完成。
舉一個(gè)簡(jiǎn)單的例子,之前的一個(gè)項(xiàng)目存在退出的功能,其具體實(shí)現(xiàn)為殺掉進(jìn)程。為了保證某些統(tǒng)計(jì)數(shù)據(jù)上報(bào)正常,不受當(dāng)前進(jìn)程退出的影響,我們可以使用獨(dú)立的進(jìn)程來(lái)完成。
多進(jìn)程的不足與缺點(diǎn)
數(shù)據(jù)共享問(wèn)題
- 由于處于不同的進(jìn)程導(dǎo)致了數(shù)據(jù)無(wú)法共享內(nèi)容,無(wú)論是static變量還是單例模式的實(shí)現(xiàn)。
- SharedPreferences 還沒(méi)有增加對(duì)多進(jìn)程的支持。
- 跨進(jìn)程共享數(shù)據(jù)可以通過(guò)Intent,Messenger,AIDL等。
SQLite容易被鎖
- 由于每個(gè)進(jìn)程可能會(huì)使用各自的SQLOpenHelper實(shí)例,如果兩個(gè)進(jìn)程同時(shí)對(duì)數(shù)據(jù)庫(kù)操作,則會(huì)發(fā)生SQLiteDatabaseLockedException等異常。
- 解決方法:可以使用ContentProvider來(lái)實(shí)現(xiàn)或者使用其他存儲(chǔ)方式。
不必要的初始化
- 多進(jìn)程之后,每個(gè)進(jìn)程在創(chuàng)建的時(shí)候,都會(huì)執(zhí)行自己的
Application.onCreate
方法。 - 通常情況下,onCreate中包含了我們很多業(yè)務(wù)相關(guān)的初始化,更重要的這其中沒(méi)有做按照進(jìn)程按需初始化,即每個(gè)進(jìn)程都會(huì)執(zhí)行全部的初始化。
- 按需初始化需要根據(jù)當(dāng)前進(jìn)程名稱,進(jìn)行最小需要的業(yè)務(wù)初始化。
- 按需初始化可以選擇簡(jiǎn)單的if else判斷,也可以結(jié)合工廠模式
一些簡(jiǎn)單的代碼示例
獲取當(dāng)前的進(jìn)程名
private String getCurrentProcessName() { String currentProcName = ""; int pid = android.os.Process.myPid(); ActivityManager manager = (ActivityManager) this.getSystemService(Context.ACTIVITY_SERVICE); for (ActivityManager.RunningAppProcessInfo processInfo : manager.getRunningAppProcesses()) { if (processInfo.pid == pid) { currentProcName = processInfo.processName; break; } } return currentProcName; }
基本的進(jìn)程初始化類
這個(gè)類用來(lái)每個(gè)進(jìn)程共用的業(yè)務(wù)初始化邏輯。
public class AppInitialization { @CallSuper public void onAppCreate(Application application) { Log.i("AppInitialization", "onAppCreate is being executed."); } }
工廠模式的應(yīng)用
public class AppInitFactory { public static AppInitialization getAppInitialization(String processName) { AppInitialization appInitialization; if (processName.endsWith(":game")) { appInitialization = new GameAppInitialization(); } else if (processName.endsWith(":music")) { appInitialization = new MusicAppInitialization(); } else { appInitialization = new AppInitialization(); } return appInitialization; } static class GameAppInitialization extends AppInitialization { @Override public void onAppCreate(Application application) { super.onAppCreate(application); Log.i("GameAppInitialization", "onAppCreate is being executed."); } } static class MusicAppInitialization extends AppInitialization { @Override public void onAppCreate(Application application) { super.onAppCreate(application); Log.i("MusicAppInitialization", "onAppCreate is being executed."); } } }
具體的調(diào)用時(shí)的代碼
public class MyApplication extends Application{ private static final String LOGTAG = "MyApplication"; @Override public void onCreate() { super.onCreate(); String currentProcessName = getCurrentProcessName(); Log.i(LOGTAG, "onCreate currentProcessName=" + currentProcessName); AppInitialization appInitialization = AppInitFactory.getAppInitialization(currentProcessName); if (appInitialization != null) { appInitialization.onAppCreate(this); } } }
是否需要多進(jìn)程
判斷是否需要多進(jìn)程,需要視具體情況而定。
內(nèi)存限制
- 研究?jī)?nèi)存占用居高不下的原因
- 如果是由內(nèi)存泄漏導(dǎo)致,嘗試解決來(lái)降低內(nèi)存占用
- 如有必要,可以通過(guò)配置largeHeap嘗試解決
除了內(nèi)存限制之外,還需要考慮是否真的需要獨(dú)立于主進(jìn)程來(lái)執(zhí)行某些操作。
關(guān)于android:process的其他問(wèn)題
在android:process
部分我們提到,如果這個(gè)屬性值以小寫字母開(kāi)頭,那么就是全局的進(jìn)程,可以被其他應(yīng)用共用。
所謂的共用,指的是不同的App的組件運(yùn)行在同一個(gè)指定的進(jìn)程中。
準(zhǔn)備條件
受制于Android系統(tǒng)的安全機(jī)制,我們需要做到以下兩個(gè)準(zhǔn)備條件才可以。
- 這個(gè)應(yīng)用使用同樣的簽名
- 兩個(gè)應(yīng)用指定同一個(gè)
android:sharedUserId
的值
具體示例
第一個(gè)App的Manifest文件,AnotherActivity運(yùn)行在名為droidyue.com的進(jìn)程中。
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.droidyue.androidmutipleprocesssample" android:sharedUserId="droidyue.com" > <application android:name=".MyApplication" android:allowBackup="true" android:icon="@mipmap/ic_launcher" android:label="@string/app_name" android:supportsRtl="true" android:theme="@style/AppTheme"> <activity android:name=".AnotherActivity" android:process="droidyue.com"/> </application> </manifest>
第二個(gè)App的Manifest文件,SecondActivity運(yùn)行在名為droidyue.com的進(jìn)程中。
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.jishuxiaoheiwu.accessfromanotherprocess" android:sharedUserId="droidyue.com" > <application android:allowBackup="true" android:icon="@mipmap/ic_launcher" android:label="@string/app_name" android:supportsRtl="true" android:theme="@style/AppTheme"> <activity android:name=".SecondActivity" android:process="droidyue.com" /> </application> </manifest>
上面的AnotherActivity和SecondActivity會(huì)運(yùn)行在一個(gè)名為droidyue.com的進(jìn)程中,盡管他們位于不同的App中。
但是這種共用進(jìn)程的方式會(huì)引發(fā)很多問(wèn)題,不太建議大家使用。
以上就是我關(guān)于Android中多進(jìn)程的一些淺顯的研究,如有問(wèn)題,歡迎指正。
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來(lái)一定的幫助,如果有疑問(wèn)大家可以留言交流。
- Android應(yīng)用開(kāi)發(fā)SharedPreferences存儲(chǔ)數(shù)據(jù)的使用方法
- Android 動(dòng)畫(huà)之TranslateAnimation應(yīng)用詳解
- Android 動(dòng)畫(huà)之ScaleAnimation應(yīng)用詳解
- Android開(kāi)發(fā)中多進(jìn)程共享數(shù)據(jù)簡(jiǎn)析
- Android 多進(jìn)程資料總結(jié)
- Android 進(jìn)程間通信實(shí)現(xiàn)原理分析
- Android應(yīng)用程序四大組件之使用AIDL如何實(shí)現(xiàn)跨進(jìn)程調(diào)用Service
- Android編程防止進(jìn)程被第三方軟件殺死的方法
- 解析后臺(tái)進(jìn)程對(duì)Android性能影響的詳解
- Android結(jié)束進(jìn)程的方法詳解
相關(guān)文章
Kotlin創(chuàng)建一個(gè)好用的協(xié)程作用域
這篇文章主要介紹了Kotlin創(chuàng)建一個(gè)好用的協(xié)程作用域,kotlin中使用協(xié)程,是一定要跟協(xié)程作用域一起配合使用的,否則可能協(xié)程的生命周期無(wú)法被準(zhǔn)確控制,造成內(nèi)存泄漏或其他問(wèn)題2022-07-07Android ProgressDialog的實(shí)例詳解
這篇文章主要介紹了Android ProgressDialog的實(shí)例詳解的相關(guān)資料,Android 開(kāi)發(fā)項(xiàng)目的時(shí)候經(jīng)常會(huì)遇到耗時(shí)的操作,這里就講下Android ProgressDialog的應(yīng)用,需要的朋友可以參考下2017-07-07Android取消EditText自動(dòng)獲取默認(rèn)焦點(diǎn)
本文主要介紹了Android取消EditText自動(dòng)獲取焦點(diǎn)默認(rèn)行為的方法,具有很好的參考價(jià)值。下面跟著小編一起來(lái)看下吧2017-03-03Android編程實(shí)現(xiàn)在Activity中操作刷新另外一個(gè)Activity數(shù)據(jù)列表的方法
這篇文章主要介紹了Android編程實(shí)現(xiàn)在Activity中操作刷新另外一個(gè)Activity數(shù)據(jù)列表的方法,結(jié)合具體實(shí)例形式分析了2種常用的Activity交互實(shí)現(xiàn)技巧,需要的朋友可以參考下2017-06-06AndroidStudio 設(shè)置格式化斷行寬度教程
這篇文章主要介紹了AndroidStudio 設(shè)置格式化斷行寬度教程,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-03-03Flutter加載圖片流程之ImageCache源碼示例解析
這篇文章主要為大家介紹了Flutter加載圖片流程之ImageCache源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-04-04