1. 背景
因?yàn)楣粳F(xiàn)在要在廣告 SDK 開發(fā)中實(shí)現(xiàn)一部分代碼的更新,或者是新增部分功能,因此需要從后臺(tái)通過網(wǎng)絡(luò)下載然后通過插件化的方式進(jìn)行加載展示。本來想的是自己實(shí)現(xiàn),但是遇到了一些問題,所以直接采用滴滴的開源框架 VirtualAPK 來實(shí)現(xiàn)。由于 VirtualAPK 是針對(duì)于 app 開發(fā)者的,所以我們?cè)诖虬臅r(shí)候需要把 SDK 稍微做一些改動(dòng)。
2. 前期準(zhǔn)備
滴滴開源框架 VirtualAPK 的 GitHub 地址為 https://github.com/didi/VirtualAPK,通過 Wiki 的內(nèi)容可以了解框架集成及使用。如果想了解原理則可以閱讀源碼進(jìn)行了解。
3. 項(xiàng)目結(jié)構(gòu)

1. 宿主結(jié)構(gòu)
內(nèi)部有 HostMainActivity、HostMyApp 兩個(gè)類
- HostMainActivity 用于插件加載和跳轉(zhuǎn)插件中的頁面
if (ContextCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED) {
String pluginPath = Environment.getExternalStorageDirectory().getAbsolutePath().concat("/testpulgin.apk");
File plugin = new File(pluginPath);
try {
PluginManager.getInstance(getApplicationContext()).loadPlugin(plugin);
} catch (Exception e) {
e.printStackTrace();
}
} else {
ActivityCompat.requestPermissions(this,
new String[]{Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.WRITE_EXTERNAL_STORAGE}, 100);
}
TextView tv_host = findViewById(R.id.tv_host);
tv_host.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent();
intent.setClassName("keven.zcdog.pluginvirtual", "keven.zcdog.pluginvirtual.PluginMainActivity");
startActivity(intent);
}
});
- HostMyApp 用于主 APP 進(jìn)行調(diào)用初始化 VirtualAPK
public class HostMyApp {
public static void init(Context context){
PluginManager.getInstance(context).init();
}
}
2. 插件結(jié)構(gòu)
插件中就只有一個(gè) Activity 和一個(gè)布局,供宿主調(diào)用。
4. 項(xiàng)目搭建
1. 項(xiàng)目配置
根據(jù) GitHub 上的配置進(jìn)行項(xiàng)目配置
- 在根目錄的 build.gradle 文件中添加
classpath 'com.didi.virtualapk:gradle:0.9.8.6'
在根目錄的 gradle.properties 文件中添加
android.useDexArchive=false
- 在宿主的 build.gradle 文件中添加
apply plugin: 'com.didi.virtualapk.host'
compile 'com.didi.virtualapk:core:0.9.8'
- 在插件項(xiàng)目的根目錄 build.gradle 文件中添加(由于我這邊插件和宿主在同一項(xiàng)目中,所以只需要配置一次即可)
classpath 'com.didi.virtualapk:gradle:0.9.8.6'
- 在插件的 build.gradle 文件中添加
apply plugin: 'com.didi.virtualapk.plugin'
virtualApk {
packageId = 0x62 // 為了避免資源沖突.
targetHost='D:/AndroidDemo/RetryVirtualApk/myaarhost' // 宿主項(xiàng)目位置,最好配置絕對(duì)路徑,避免出錯(cuò).
applyHostMapping = true // 默認(rèn)為 true.
}
2. 生成插件包
先在 myaarhost 的 build.gradle 文件中進(jìn)行修改
將
apply plugin: 'com.android.library'
替換為
apply plugin: 'com.android.application'
然后點(diǎn)擊 Android Studio 右側(cè)工具欄對(duì)宿主進(jìn)行 build

然后點(diǎn)擊插件項(xiàng)目的 按鈕生成插件 apk

如果可以成功,可以在插件項(xiàng)目中找到(由于生成的是 release 包,所以需要配置下簽名)

如果過程中報(bào)錯(cuò),可參考 VirtualAPK 踩坑實(shí)錄 ,我當(dāng)時(shí)就是借鑒這篇文章的一些解決方法解決的。
5. 測(cè)試
- 將 myaarhost 改回 library,然后生成 aar
- 將產(chǎn)生的插件 apk 放入到手機(jī)根目錄
- 新建項(xiàng)目,依賴 aar 包
- 在新建項(xiàng)目根目錄配置
classpath 'com.didi.virtualapk:gradle:0.9.8.6'
在新建項(xiàng)目 app 下配置
compile 'com.didi.virtualapk:core:0.9.8'
- 新建類繼承 Application ,調(diào)用 aar 中的方法初始化 VirtualAPK
public class MyApplication extends Application {
@Override
protected void attachBaseContext(Context base) {
super.attachBaseContext(base);
HostMyApp.init(base);
}
}
- 啟動(dòng) aar 中的 Activity
Intent intent = new Intent(MainActivity.this, HostMainActivity.class);
startActivity(intent);
- 實(shí)現(xiàn)效果

6. 全部代碼
本文所有代碼見 GitHub 地址
https://github.com/keven0632/VirtualAarDemo