博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
类加载器ClassLoader(一)动态加载
阅读量:6207 次
发布时间:2019-06-21

本文共 3047 字,大约阅读时间需要 10 分钟。

前段时间碰了下插件化,然后就简单的了解下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方法,最后看结果

img_08e721286f94fb6466ba71d3a1a55389.png

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

转载地址:http://srzja.baihongyu.com/

你可能感兴趣的文章
Android消息机制
查看>>
《重构---改善既有代码的设计》之详解一
查看>>
OpenWrt:WRTnode2P实现联网
查看>>
识别二维码
查看>>
Google Play 购买(IAB)测试流程
查看>>
深入理解行内元素的布局
查看>>
D3.js绘制实时映射的缩略图
查看>>
记一次Oracle控制文件分析情况
查看>>
MyBatis ResultType使用自定义TypeHandler
查看>>
翻译 | 如何规模化 React 应用
查看>>
[云框架]SMACK大数据架构
查看>>
你想要的——redux源码分析
查看>>
Kafka团队修改KSQL开源许可,怒怼云厂商
查看>>
Eclipse基金会发布MicroProfile 2.2,适用于Java微服务
查看>>
QCon上海2015 架构实战热点前瞻
查看>>
JetBrains宣布Project Rider:一款C#跨平台IDE
查看>>
Lodash,你正在使用的JavaScript库
查看>>
苏宁11.11:一种基于神经网络的智能商品税分类系统
查看>>
C#的未来:闭包注解
查看>>
解读2015之Golang篇:Golang的全迸发时代
查看>>