前言
- 需求:APK A 加载 APK B的代码,调用APK B提供的接口实现相关功能。
- 实现技术:通过构建对应包名的classloader加载指定类,并调用相关方法。
- 好处:主APK A 和 APK B在一定程度上解耦,双方的接口定义好了后,无需APK A升级,通过升级APK B及可事项相关功能的迭代。
- 运用场景:APK A需要跟随固件升级(即OTA), APK B可独立升级,将需要频繁迭代且相对独立的功能放到可独立升级的APK B中,采用Classloader的方式可满足频繁迭代的诉求,APK B就相当于 APK A的一个插件,通过升级插件即可升级 APK A对应的功能。
代码实现
1:建立APK B的工程,其包名为com.apkb.test
2:新建 TestB 类,代码如下:
package com.apkb.test;
import android.content.Context;
import android.util.Log;
public class TestB {
private static final String TAG = "TestB";
public TestB(){
}
public void init(Context context) {
Log.d(TAG,"init context ="+context);
}
public void onDestroy() {
Log.d(TAG,"onDestroy");
}
}
3:新建APK A的工程
新建MainActivity,代码如下,关键部分见loadApkBClassObj方法的注释
package com.apka.test;
import android.app.Activity;
import android.content.Context;
import android.os.Bundle;
import java.lang.reflect.Method;
public class MainActivity extends Activity {
private static final String APK_B_CLS_NAME = "com.apkb.test.TestB";
private static final String APK_B_PKG = "com.apkb.test";
private Object mInterface = null;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
loadApkBClassObj();
}
//创建 Apk B中指定类的实例
private void loadApkBClassObj() {
try {
//构建APK B对应的包名的Context
Context packageContext = this.getApplicationContext().
createPackageContext(APK_B_PKG,
Context.CONTEXT_INCLUDE_CODE | Context.CONTEXT_IGNORE_SECURITY);
//通过packageContext拿到对应的classloader
ClassLoader pkgClassloader = packageContext.getClassLoader();
//加载指定的类
Class<?> cls = pkgClassloader.loadClass(APK_B_CLS_NAME);
//通过newInstance实例化指定类的对象
mInterface = cls.newInstance();
init();
} catch (Throwable e){
}
}
//调用 APK B中的 TestB 对应的初始化方法
private void init() {
try {
Method method = mInterface.getClass().getMethod("init", Context.class);
method.invoke(mInterface, this.getApplicationContext());
} catch (Exception e) {
e.printStackTrace();
}
}
//调用 APK B中的 TestB 对应的onDestroy方法
private void doDestroy() {
try {
Method method = mInterface.getClass().getMethod("onDestroy");
method.invoke(mInterface);
mInterface = null;
} catch (Exception e) {
e.printStackTrace();
}
}
@Override
protected void onDestroy() {
super.onDestroy();
doDestroy();
}
}
4:安装APK B,无需运行,安装APK A,打开APK A,可查看到如下日志,APK A成功的加载到了APK B中的指定类,并成功调用对应方法。可查看到如下日志:
注意点:加载的APK B中类不可被混淆
finish~