參考
Android多進(jìn)程的數(shù)據(jù)庫訪問問題
Android中多進(jìn)程的應(yīng)用
Android IM 開發(fā)小結(jié)
關(guān)于 Android 進(jìn)程毖爻蓿活君账,你所需要知道的一切
在Android開發(fā)中座柱,我們可能會使用單獨(dú)的進(jìn)程來做一些事情焕梅,比如推送服務(wù),心跳服務(wù)等担孔,這些不需要主應(yīng)用啟動江锨,只需要一個獨(dú)立的進(jìn)程即可。這時候我們一般都會采用啟動一個后臺Service攒磨,這個Service運(yùn)行在一個獨(dú)立的進(jìn)程中,比如在Androidmainfest.xml中配置Service的android:process=”:push”指定該Service運(yùn)行在:push進(jìn)程中汤徽。
一般情況下這樣做是沒有任何問題的娩缰,但是如果你在你的Application的onCreate方法里有對數(shù)據(jù)庫的操作,或者該Service里有對數(shù)據(jù)庫的操作谒府,就會有可能會有兩個進(jìn)程同時操作一個數(shù)據(jù)庫的情況拼坎,一個是你的主應(yīng)用的進(jìn)程,進(jìn)程名為你的包名例如org.flysnow;一個是你的Service所在的進(jìn)程完疫,進(jìn)程名為你的包名+”:push”,即org.flysnow:push泰鸡。這兩個進(jìn)程可能在同一時間訪問同一個數(shù)據(jù),同一個配置文件等壳鹤。這就可能造成資源的競爭訪問盛龄,造成的問題就不可預(yù)料了,比如數(shù)據(jù)庫損壞,數(shù)據(jù)丟失等余舶。
多進(jìn)程其實和多線程一樣啊鸭,并發(fā)訪問的時候產(chǎn)生的問題很難預(yù)料,在多線程的時候我們有鎖等機(jī)制控制資源的訪問匿值,但是在多進(jìn)程中比較難赠制,雖然有文件鎖、排隊等機(jī)制挟憔,但是在Android里很難實現(xiàn)钟些,畢竟在Android里一個進(jìn)程就是一個VM虛擬機(jī),底層的東西控制不了绊谭,Java層又沒有辦法控制政恍,所以在多進(jìn)程中一定不要有并發(fā)增刪改文件的操作。
解決問題的核心就是不并發(fā)訪問同一個文件龙誊,多線程時就使用Lock機(jī)制抚垃;多進(jìn)程的時候就避免進(jìn)行數(shù)據(jù)庫的訪問,比如只做一些心跳趟大、激活鹤树、消息抓取等操作,涉及到把消息存儲到數(shù)據(jù)庫逊朽,訪問配置文件等操作還是調(diào)用主進(jìn)程進(jìn)程操作罕伯。還一個要注意的就是Application的oncreate方法里要避免多進(jìn)程訪問同一文件,因為沒一個進(jìn)程初始化都會執(zhí)行該方法叽讳,可以在一些進(jìn)程初始化的時候不需要文件操作的時候不要進(jìn)行文件操作追他,比如在onCreate里獲取當(dāng)前的進(jìn)程,不等于包名的就不進(jìn)行文件的訪問操作岛蚤,獲取進(jìn)程可以使用android.os.Process.myUid()方法邑狸。
好處:
1、分擔(dān)主進(jìn)程的內(nèi)存壓力涤妒。我們的應(yīng)用越做越大单雾,內(nèi)存越來越多,將一些獨(dú)立的組件放到不同的進(jìn)程她紫,它就不占用主進(jìn)程的內(nèi)存空間了硅堆。比如在啟動一個不可見的輕量級私有進(jìn)程,在后臺收發(fā)消息贿讹,或者做一些耗時的事情渐逃,或者開機(jī)啟動這個進(jìn)程等。
2民褂、防止主進(jìn)程被殺守護(hù)進(jìn)程茄菊,守護(hù)進(jìn)程和主進(jìn)程之間相互監(jiān)視疯潭,有一方被殺就重新啟動它。
壞處:
1买羞、多占了系統(tǒng)的內(nèi)存空間袁勺,很容易沾滿而導(dǎo)致卡頓,同時也消耗用戶的電量畜普。同時在啟動單獨(dú)進(jìn)程時期丰,進(jìn)程的創(chuàng)建會影響繼承Application的實例,onCreate()會再次執(zhí)行一遍吃挑。
2钝荡、不同進(jìn)程之間內(nèi)存不能共享,最大的弊端是他們之間通信麻煩舶衬,不能將公用數(shù)據(jù)放在Application中埠通,堆棧信息、文件操作也是獨(dú)立的逛犹,如果他們之間傳遞的數(shù)據(jù)不大并且是可序列化的端辱,可以考慮通過Bundle傳遞, 如果數(shù)據(jù)量較大虽画,則需要通過AIDL或者文件操作來實現(xiàn)舞蔽。