类加载器ClassLoader(一)动态加载

前段时间碰了下插件化,然后就简单的了解下classloader。虽然框架是能用,但是还是打算使用classloader来实现动态加载。

功能:主要开发个插件apk,里面有打印信息,然后宿主导入apk再用classloader加载出相应的类,并且调用里面的方法。

一. java的ClassLoader

1.classloader类型

先了解下java的ClassLoader,因为android的ClassLoader会有些不同。
java默认提供三种ClassLoader
(1)Bootstrp ClassLoader
(2)ExtClassLoader
(3)AppClassLoader
先记住有这三种就行,至于有什么用,就先不讲了,因为我们用到的是android的ClassLoader。还有注意的是Bootstrp ClassLoader是底层用C++写的,它们的关系是AppClassLoader继承ExtClassLoader,ExtClassLoader继承Bootstrp ClassLoader。

2.classloader工作流程

java的classloader有个双亲委派模式,网上有很多,看别人的图就能比较好懂,我这边就不盗图了。简单来说就是当需要加载一个类的时候,先去看AppClassLoader有没有加载过,没有再看ExtClassLoader有没有加载过,没有再看Bootstrp ClassLoader有没有加载过,没有再从Bootstrp ClassLoader中找是不是他那边的类,不是再向下交给ExtClassLoader处理。
双亲委派这个要记住。然后双亲委派的作用主要是能让类不重复加载和保证安全(比如你无法自定义系统类)。

二. android的ClassLoader

大概了解java的ClassLoader之后我们也来简单了解下Android的ClassLoader。

1.classloader类型

android的classloader就比较多了,没用到的后面再讲,这里我们要做动态加载,主要也是用到3个classloader
(1)ClassLoader 这是顶层的classloader类
(2)BaseDexClassLoader 这个类主要做类加载的操作
(3)PathClassLoader 这个是BaseDexClassLoader 的子类
(4)DexClassLoader 这个是BaseDexClassLoader 的子类
然后这样,BaseDexClassLoader 是做具体的操作,我打算放到后面的篇章再详细说,所以要做动态加载主要看PathClassLoader和DexClassLoader两个类。
说得简单一点,这两个类的最好区分就在构造方法。
(1)先看DexClassLoader的构造方法

public DexClassLoader(String dexPath, String optimizedDirectory, String librarySearchPath, ClassLoader parent) {
        super(dexPath, new File(optimizedDirectory), librarySearchPath, parent);
}

dexPath是Dex的路径,optimizedDirectory指的是优化之后的目录,有个说法是装生成的odex文件。librarySearchPath指要使用的C/C++代码,parent指父加载器。
(2)再看PathClassLoader的构造方法

public class PathClassLoader extends BaseDexClassLoader {
    public PathClassLoader(String dexPath, ClassLoader parent) {
        super(dexPath, null, null, parent);
    }
    public PathClassLoader(String dexPath, String librarySearchPath, ClassLoader parent) {
        super(dexPath, null, librarySearchPath, parent);
    }
}

看得出PathClassLoader只有三个构造方法,对比DexClassLoader,少了optimizedDirectory。
所以说dalvik虚拟机上PathClassLoader只能加载已安装apk的dex。而我们要实现动态加载,所以使用的类加载器是DexClassLoader

2.classloader工作流程

和java一样,也是使用双亲委派

三. 写个动态加载的Demo

我们先搞个简单的,不含任何资源,就仅仅调用Java代码,确认是否能够跑一遍流程。

1.开发插件
public class TestLog {

    public void say(String str){
        Log.v("mmp","插件打印:"+str);
    }

}

我就加了这个一个类,然后打包成apk

2.宿主动态加载
    private void loadPlugin(){
        try {
            String pluginPath = getExternalFilesDir("plugin").getPath() + "/TestPlugin.apk" ;
            String targetPath = getExternalFilesDir("target").getPath();
            DexClassLoader dexClassLoader = new DexClassLoader(pluginPath,targetPath,null,getClassLoader());
            Class<?> cls = dexClassLoader.loadClass("com.example.kylin.testapp.TestLog");
            cls.getDeclaredMethod("say",String.class).invoke(cls.newInstance(),"AAAAAAAAAAAAA");
        } catch (Exception e) {
            e.printStackTrace();
            Log.v("mmp","错误:"+e.toString());
        }
    }

简单就这样写,我们把插件打出的apk命名为TestPlugin.apk,然后放到plugin文件夹下,调用DexClassLoader 去获取到Class,再用反射调用Class的say方法,最后看结果



确实是有打印的,这样我们就相当于能正常跑完一个动态加载类的流程。但是这个地方还是有些问题,运行之后我看了下target文件夹是没东西的,而odex文件是在plugin文件夹下生成了一个oat文件夹里面,这就比较奇怪,我先看看什么原因再补上。


四. (补充)加载资源

除了用ClassLoader加载代码,比较核心的操作还有资源和生命周期,生命周期这个涉及的水有点深,我这边就打算把加载资源也放在一起写算了,不打算再分一章了。

要加载插件的资源,要用到的类是AssetManager,当创建Resources的时候第一个参数就是传AssetManager。

这里我写个Demo获取插件apk中的一张图片,然后用一个ImageView来展示。

        // 加载图片
        try {
            AssetManager assetManager = AssetManager.class.newInstance();
            int path = (int) AssetManager.class.getDeclaredMethod("addAssetPath",String.class).invoke(assetManager, pluginPath);
            Resources resources = new Resources(assetManager, getResources().getDisplayMetrics(), getResources().getConfiguration());
            Drawable drawable = resources.getDrawable(resources.getIdentifier("test", "drawable","com.example.plugin.testapp"));
        }catch (Exception e){
            Log.v("mmp","加载图片失败"+e.toString());
        }

resources.getIdentifier就相当于R.,这个没啥好说的,要注意的是第三个参数传的是插件的包名。

比较需要注意的是addAssetPath这个方法,这个是AssetManager 根据路径获取资源的方法,不过是一个隐藏方法(@hide)所以需要用反射来调用,但是android9.0之后谷歌打算禁用 hide APIs了,这个问题以后碰到再说吧,暂时注意一下就行。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,390评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,821评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,632评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,170评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,033评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,098评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,511评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,204评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,479评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,572评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,341评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,893评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,171评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,486评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,676评论 2 335

推荐阅读更多精彩内容