有人问我是否支持加载插件中的资源,这个当然是支持的。实现也比较简单。
加载资源文件
另外一个必须解决的问题是加载插件中的资源,包括布局文件和图片等。这个需要调用系统的AssetManager来完成,addAssetPath是个私有方法,所以使用反射来调用。示例代码如下:
AssetManager am = (AssetManager) AssetManager.class.newInstance();
am.getClass().getMethod("addAssetPath", String.class)
.invoke(am, dexPath);
Resources ctxres = context.getResources();
Resources res = new Resources(am, ctxres.getDisplayMetrics(), ctxres.getConfiguration());
然后在Activity初始化的时候把这个新建的res对象设置给activity:
Reflection.setField(activity, "mResources", contextHook.plugin.resources);
这样插件中就能自动加载自身apk中的资源。如果宿主拿到这个Res对象,也能访问插件的资源。
Class ref in pre-verified异常
在Android4.4以下版本才会出现这个异常。后来才知道是因为在旧版本中虚拟机中,如果一个类没有引用其它dex中的类,就会被打上CLASS_ISPREVERIFIED标记。而事实上通过重载ClassLoader机制,实际引用的类来自其它dex,就会出现这个异常。详情见这里
有两种解决办法:
- 类内引用一个外部的类。在所有类构造函数中插入如下代码: if (ClassVerifier.PREVENT_VERIFY) {System.out.println(AntilazyLoad.class);}
- 插件中打包时不要包含宿主中已有的类(ADT中JavaBuildPath中的Order and Export把勾去掉。AndroidStudio中把compile改为provided即可。)