最新依賴-jitpack
Google的AAB出來(lái)好久了厌蔽,但在國(guó)內(nèi)一直沒(méi)法用。知道Qigsaw后就開始學(xué)習(xí)吱肌,弄了好久都沒(méi)學(xué)會(huì)篇梭,太煩了中途都放棄了看蚜。后來(lái)有時(shí)間了,又來(lái)學(xué)習(xí)這個(gè)甩鳄,因?yàn)檎鎸?shí)用俺讯取!
直到學(xué)會(huì)了妙啃,還有問(wèn)題
- 編譯后無(wú)法Clean
- 不支持最新的 'com.android.tools.build:gradle:4.1.3'
- Demo代碼太繁瑣第晰,不容易懂
- 使用起來(lái)也比較麻煩
學(xué)會(huì)后,我覺得不應(yīng)該這樣麻煩,是可以非常簡(jiǎn)單的茁瘦!
可以把Qigsaw拆分為運(yùn)行框架和打包工具品抽,因此有了QigsawBundle。
DynamicProvider
Provider在應(yīng)用啟動(dòng)時(shí)就會(huì)被調(diào)用甜熔,Dynamic中的Provider第一次啟動(dòng)是找不到的圆恤,應(yīng)用會(huì)直接報(bào)錯(cuò),無(wú)法啟動(dòng)腔稀。Qigsaw為每個(gè)DynamicProvider生成了一個(gè)裝飾類盆昙,找不到原DynamicProvider時(shí),就調(diào)用裝飾類焊虏。
QigsawBundle不生成任何裝飾類淡喜,而是通過(guò)工具DynamicProviderSwitch自動(dòng)把DynamicProvider設(shè)置為關(guān)閉狀態(tài)(android:enabled="false")。啟動(dòng)應(yīng)用時(shí)诵闭,再讀取manifest中哪些DynamicProvider類存在炼团,如果存在則啟動(dòng)Provider,不存在則忽略疏尿。Split安裝后重試未啟動(dòng)成功的DynamicProvider瘟芝。
DynamicProviderSwitch:編譯時(shí)自動(dòng)關(guān)閉Split中的Provider,應(yīng)用啟動(dòng)時(shí)褥琐,啟動(dòng)存在的DynamicProvider锌俱。
DynamicProviderSwitch支持Google的AAB,可以直接用在海外版的App中
Activity/Service/BroadcastReceiver
Qigsaw打包時(shí)敌呈,織入了一些代碼贸宏。經(jīng)測(cè)試證明,不需要在編譯期間織入代碼的磕洪,
Activity/Service/BroadcastReceiver都可以通過(guò)運(yùn)行時(shí)代碼解決锚赤。
可以在ActivityLifecycleCallbacks.onActivityPreCreated中l(wèi)oadResources,
另外兩個(gè)使用的application.resources 褐鸥,所以不需要處理。
以下代碼都測(cè)試通過(guò)了赐稽。
override fun onActivityPreCreated(activity: Activity, savedInstanceState: Bundle?) {
SplitInstallHelper.loadResources(activity, activity.resources)
}
Service.resources ==applicationContext.resources//true
BroadcastReceiver.context.resources==context.applicationContext.resources//true
onActivityPreCreated 只有Android10+才支持叫榕,之前的版本沒(méi)有用。所以需要用織入代碼的方式來(lái)解決姊舵。
QigsawBundle在onActivityPreCreated中注入resources晰绎,因此省去了編譯期織入代碼。
private Qigsaw(Context context, Downloader downloader, @NonNull SplitConfiguration splitConfiguration) {
this.context = context;
this.downloader = downloader;
this.splitConfiguration = splitConfiguration;
this.currentProcessName = ProcessUtil.getProcessName(context);
this.isMainProcess = context.getPackageName().equals(currentProcessName);
InjectActivityResource.inject((Application) context);//QigsawBundle注入resources
}
CompatBundle
為了讓獨(dú)立打包的Split能運(yùn)行起來(lái)括丁,需要Qigsaw框架做一些兼容處理荞下。如果不實(shí)現(xiàn)以下接口,則完全按照標(biāo)準(zhǔn)的Qigsaw方式運(yùn)行。具體使用參考Demo尖昏。
public interface ICompatBundle {
/**
* for 'parseSplitContentsForDefaultVersion'
*/
@Nullable
String readDefaultSplitVersionContent(@NonNull Context context, @NonNull String fileName);
@NonNull
String getMD5(@NonNull File file);
@NonNull
String getMD5(@NonNull InputStream inputStream);
/**
* 在onActivityPreCreated中注入resources
*/
boolean injectActivityResource();
/**
* 因?yàn)闆](méi)有生成任何裝飾類仰税,所以ComponentInfoManager是沒(méi)有數(shù)據(jù)的,所以需要禁用了
*/
boolean disableComponentInfoManager();
/**
* 沒(méi)有自動(dòng)生成的 qigsawConfig抽诉,需要指定一個(gè)自己創(chuàng)建的
*/
Class<?> qigsawConfigClass();
}
ApkMd5
QigsawBundle中所有Apk計(jì)算MD5都使用的ApkMd5陨簇,源碼也在項(xiàng)目中。
ApkMd5:"AndroidManifest.xml"去掉版本號(hào)迹淌,去掉"META-INF/BNDLTOOL.RSA"河绽、 "META-INF/BNDLTOOL.SF"、"META-INF/MANIFEST.MF"唉窃,所有文件排序耙饰,再計(jì)算APK的MD5。
QigsawBundle可以做到:只修改Base的版本號(hào)(versionName&versionCode)纹份,然后打包苟跪。所有Split的ApkMd5值都是固定的,僅Base包變了矮嫉。proguard存在一個(gè)BUG削咆,需要特別方法才能做到這樣。具體請(qǐng)參考 穩(wěn)定混淆App
DEMO
有時(shí)間了會(huì)寫個(gè)QigsawBundle使用介紹蠢笋,先寫了個(gè)原理拨齐。
DEMO
最后
如果本文幫助到了你,也幫我點(diǎn)個(gè)贊吧昨寞!
如果你愿意瞻惋,還可以贊賞一杯咖啡或一瓶水,非常感覺你的慷慨援岩!