QigsawBundle技術(shù)細(xì)節(jié)

最新依賴-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é)你的慷慨!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容