以上准备工作 做好了之后, 第二步就开始撸DexClassLoader的代码了。
第二步:直接贴上代码,利用java 类加载
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}
//动态加载dex文件 这里是个jar 包里面的方法
public void start(View view) {
//dex解压释放后的目录
final File dexOutPutDir = getDir(“dex”, 0);
//dex所在目录
final String dexPath = Environment.getExternalStorageDirectory().toString() + File.separator + “new_log.jar”;
//第一个参数:是dex压缩文件的路径
//第二个参数:是dex解压缩后存放的目录
//第三个参数:是C/C++依赖的本地库文件目录,可以为null
//第四个参数:是上一级的类加载器
DexClassLoader classLoader = new DexClassLoader(dexPath, dexOutPutDir.getAbsolutePath(), null, getClassLoader());
try {
final Class> loadClazz = classLoader.loadClass(“plugin.charles.com.logutil.LogUitl”);
final Object o = loadClazz.newInstance();
final Method printLogMethod = loadClazz.getDeclaredMethod(“printLog”);
printLogMethod.setAccessible(true);
printLogMethod.invoke(o);
} catch (Exception e) {
e.printStackTrace();
}
}
}
下面介绍的是动态加载未安装的apk.然后读取apk中图片,显示在宿主app中
1.首先准备换肤的apk
skindemo-debug.apk
skindemo2-debug.apk
这2个apk中都有一张图片,然后我们push这两个apk到sdcard中就可以
资源加载问题,怎么通过Resources 加载另外一个apk中的图片
/**
* 获取AssetManager 用来加载插件资源
*
* @param pFilePath 插件的路径
* @return
*/
private AssetManager createAssetManager(String pFilePath) {
try {
final AssetManager assetManager = AssetManager.class.newInstance();
final Class> assetManagerClazz = Class.forName(“android.content.res.AssetManager”);
final Method addAssetPathMethod = assetManagerClazz.getDeclaredMethod(“addAssetPath”, String.class);
addAssetPathMethod.setAccessible(true);
addAssetPathMethod.invoke(assetManager, pFilePath);
return assetManager;
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
//这个Resources就可以加载非宿主apk中的资源
private Resources createResources(String pFilePath) {
final AssetManager assetManager = createAssetManager(pFilePath);
Resources superRes = this.getResources();
return new Resources(assetManager, superRes.getDisplayMetrics(), superRes.getConfiguration());
}
3.宿主apk,动态换肤的点击事件代码
private boolean mChange = false;
private String skinType = “”;
private int APK_TYPE;
/**
* 替换bg
*/
public void replace(View view) {
if (!mChange) {
skinType = “skindemo-debug.apk”;
APK_TYPE = 1;
mChange = true;
} else {
skinType = “skindemo2-debug.apk”;
APK_TYPE = 2;
mChange = false;
}
final String path = Environment.getExternalStorageDirectory() + File.separator + skinType;
final String pkgName = getUninstallApkPkgName(this, path);
dynamicLoadApk(path, pkgName);
}
private void dynamicLoadApk(String pApkFilePath, String pApkPacketName) {
File file = getDir(“dex”, Context.MODE_PRIVATE);
//第一个参数:是dex压缩文件的路径
//第二个参数:是dex解压缩后存放的目录
//第三个参数:是C/C++依赖的本地库文件目录,可以为null