在開發Replugin插件時候遇到了很多的坑,當初都整理成了文檔,現在年底正好有空閑時間,我都整理出來,看看小伙伴們都有遇到的沒有:
一、接入插件存在問題
1、第一次啟動插件過慢。
2、如果安裝沒有啟動,就直接卸載,有時候會報 ANR(偶爾)
3、安裝外部插件:必須為apk的絕對路徑。
4、插件中使用SO庫,需要在主程序中添加,對應的依賴庫。
5、android7.0共享權限等,需要在主程序的Manifest中添加。
6、插件中用到的權限,需要在主程序Manifest中注冊。
7、插件中主程的公共類需要以jar的形式在各插件中使用。
8、在插件中使用第三方SDK,會存在一些坑,(第三方的sdk與使用插件會存在沖突,需要在開發中注意)
9、當我們插件數量達到20多個時,會發現隨著插件的增多,“資源分區ID”會越來越難管控,這對我們來說,也是一種挑戰。
10、 插件中使用隱式Intent進行跳轉。
這是坑位導致的。只能接受隱式intent,不接受顯示intent
意圖只能接受隱式調用。
二、360wiki中存在的問題
1.我們支持插件間,插件和宿主間的資源交互,但不支持(也不打算支持,2013年踩了無數坑,后述)“直接使用資源”的方式。即便支持這一特性會減少改動量。
這里所說的“直接使用資源”是指:插件A通過?!癛.drawable.common_xxx”來使用插件B或主程序的資源。
2.插件中使用高德地圖的經驗分享?
關于高德地圖的key值,請用宿主的包名創建一個key值在插件的AndroidManifest.xml中,否則會顯示key值不正確。
解釋:高德地圖的key值,高德地圖是用宿主的包名校驗的,插件的包名沒有校驗。
3.請一定要確保符合Gradle開發規范,也即“必須將包名寫在applicatonId”,而非AndroidManifest.xml中(通常從Eclipse遷移過來的項目可能出現此問題)。如果不這么寫,則有可能導致運行時出現“Failed to find provider info for com.ss.android.auto.loader.p.main”的問題。具體可參見 #87 Issue的問答。
4.請將apply plugin: 'replugin-host-gradle'放在 android{} 塊之后,防止出現無法讀取applicationId,導致生成的坑位出現異常。
5.如果您的應用需要支持AppComat,則除了在主程序中引入AppComat-v7包以外,還需要在主程序的build.gradle中添加下面的代碼若不支持AppComat則請不要設置此項:
repluginHostConfig {
useAppCompat = true
}
6.無法啟動插件activity:log提示java.lang.NoClassDefFoundError: library.d 或 java.lang.NoClassDefFoundError: com.qihoo360.replugin.Entry$1
原因:插件過大,導致使用了mutidex的處理,而RePlugin的包恰好被分在class2.dex,然后就拋出找不到replugin相關類的異常,導致插件加載失敗
解決方案:1.插件的代碼盡量小,盡量保證在一個dex中;2.可以在gradle中,指定replugin插件庫的類分配在主dex中。
7.還有一種可能,應該是插件被“分包”導致的問題,打包后雖然有主classes有找到com.qihoo360.replugin.Entry但還是報錯,自己手動添加maindexlist.txt在里面添加com.qihoo360.replugin打包進去就好了,具體操作請參考http://blog.csdn.net/gaozhan_csdn/article/details/52024497
三、replugin插件不足之處
1、replugin插件中的api不全
2、不同的版本可能不兼容
3、v7包
插件布局加載后莫名其妙的多了一些外邊距,無法全屏獨立運行是正常
答:
插件不能共享資源宿主的v7包,注意如果你違反這條去嘗試開發插件。會導致你的插件apk布局莫名其妙的多出一些邊距。
注意:
即便插件中你沒有依賴v7包,也是可以使用AppCompatActivity的。但是包里不會包含v7包的資源。
是因為SdkVersion版本過高所以可以直接使用。解決辦法:
在插件中單獨依賴一份v7包,而不是共享宿主的。
4、插件橫屏
插件目前manifest配置了橫屏是無效的。這也許是坑位導致的,可以動態在onCreate的時候改變一下
??protected void onCreate(Bundle savedInstanceState) {
????????super.onCreate(savedInstanceState);
????????if (getRequestedOrientation() != ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE) {
????????????setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE);
????????}
??????????·····
}
5、熱更新插件
wiki文檔里已經講過了,對于沒有啟動(加載過)的插件是可以安裝后就運行的。但是若插件被啟動過或正在運行時則不行。針對特殊的應用場景不得不重啟app,而且還是主動重啟。
那么發現如果開啟了persistentEnable = true常駐應用內存。(就是一個服務)會導致即便重啟了app,插件安裝后還是沒有被更新。這里迫使我關掉了persistentEnable選項,隨后重啟后就成功更新了。注意我這里說的重啟是非root下,應用自己殺死自己,而非用戶主動kill掉。
6、幽靈魔術
什么叫幽靈魔術問題?就和變魔術一樣,我好像沒有改什么邏輯性的代碼,突然bug又好了。
多半是你動了rePlugin中的gradle里的配置項,可能留下了上一次的緩存在里面,你要做的就是每當改動了gradle里的配置最好搭配Clean Project一下。
四、RePlugin目前不支持以下功能
1、Application和主程序相關
·插件權限聲明無效,只認主程序的
o建議:將真正需要用到的權限提前在“主程序”中聲明好。
o原因:權限部分是系統在安裝主程序時,會讀取其AndroidManifest.xml中的權限部分,并放入系統的package.xml中,除了系統自己的核心應用(和Root)以外,外界根本無法修改。目前已知的插件化、熱更新方案均不支持“動態權限的增刪改”。
·插件聲明Target SDK無效,只認主程序的
o建議:根據需要來調整主程序的Target SDK,同時插件和主程序最好同時調整。這樣即便插件變成單品,也能做到“天然的適配”。
o原因同上,不贅述
·不支持“雙開應用”功能。也即:“不經修改,直接將任意APK放入RePlugin進行‘免安裝運行’”
o建議:RePlugin的核心訴求是在“穩定(1 Hook,0 Binder Hook)的前提下”,盡可能保持靈活,且以最小的成本,產生更多的插件,而非“免修改APK直接運行”。
o如確實需要開發雙開、免安裝應用,或公司接入的插件均“只提供APK(不含混淆后的AAR)”的,可使用我們360的另一套成熟的框架:DroidPlugin,或 @Lody 羅迪的?VirtualApp。
o原理:要想實現這種效果,必須先Hook AMS、PackageManager,甚至還要做很多Native Hook的工作。Hook點會很多,也會有不少的適配工作。但只要Hook齊了,就能讓一個應用“直接運行起來”,但代價是“適配量太大”,對未來新增的機型充滿了“未知性”。
·不支持Notification的Layout資源
o建議:可在主程序中預埋一些Layout模板,插件可以套用此模板。另外,“圖片資源”以及文案等,是可以通過Drawable和String來透傳的,是完美支持的。
o原因:RemoteViews的性質決定的。
2、交互
·不支持“宿主和主程序無縫使用資源”(共享資源)方案
o建議:采用更穩定的一些API來“迂回的”獲取資源。順便說一句,RePlugin支持“共享View”方案(例如公共UI庫的使用等),這樣“無縫獲取資源”的場景,在其它方面也能得到滿足。
o原因:這是RePlugin的性質所決定的。也即——盡可能的穩定,不要有適配代碼。因篇幅所限,請點擊此處閱讀《FAQ》了解這塊兒的更多內容。
·不支持“主程序退出后”的靜態廣播
o建議:請盡量避免過分依賴“主程序停止后”的“靜態廣播”(Android 8.0以上更是如此,因為原生就禁用了)。此外,只要常駐進程存在,則靜態廣播將一直生效。
o原因:畢竟是“模擬的”靜態廣播,本質上仍是“動態注冊廣播”。這和系統直接喚起應廣播所在進程,然后發送的“靜態廣播”是不一樣的。
3、Activity
·overridePendingTransition,僅支持使用宿主和系統的
o建議:將動畫資源“預埋在主程序里”,且確保其ID為固定(利用public.xml),這樣可以通過拿主程序的動畫ID來傳遞給系統,實現相應效果。目前手機衛士等產品都是這樣做的,效果不錯。
o原因:此方法僅會傳給AMS兩個int值,且由AMS層進行資源解析并實現動畫效果,根本到不了客戶端。目前市面上所有插件化、熱更新方案均無法實現此功能。
五、rePlugin插件使用注意事項:
1.權限和SO庫, 必須在主宿,添加一份,程序才能正常運行。
2.7.0系統,拍照共享路徑必須在androidManFist.xml中添加。
3.注意mavenLocal()必須放在jcenter()之前。?
4.判斷插件是否安裝: 包名。
5.卸載插件,默認使用包名進行卸載,也可使用別名進行卸載。
6.定義的別名必須唯一,要不安裝不上(兩個相同的別名只能安裝1個)。