最新依賴-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ì)后,我覺(jué)得不應(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è)贊吧!
如果你愿意,還可以贊賞一杯咖啡或一瓶水,非常感覺(jué)你的慷慨!