Android-ClassLoader

ClassLoader

Introduction

  • Java是半编译半解释语言,任何一个.java文件(其实就是一个类文件)都要被jdk的编译器编译成class文件(Android上会把所有的class文件进行一个合并,并对class文件做一些优化,比如去除运行时常亮池里重复的部分),class文件就是jvm的可执行文件,任何一个class文件都必须要被JVM加载(就是一次IO,把class文件读入内存,然后JVM再去进行相应的处理)后才能在运行时使用,一个class文件的加载还分几个步骤:
    • 装载
      • 等于一次IO
    • 校验
      • 任何一个文件包括可执行文件、class文件都是可以人为修改或者编写的(只是可执行文件和class文件等我们都是用编译器来自动生成的),所以就可能有一些不合法的指令
    • 准备
      • 为静态变量分配空间(加载的是类文件,又没有对象,所以当然只能是静态变量了),只是分配,没有赋值动作
    • 解析
      • 官方说法:把符号引用解析为直接引用,简单且非正式的解释一下,在class文件里,java类的每一个方法和变量都是依靠字符串的形式存储的,而不管怎么样,我们最后肯定是要像C/C++一样拿到方法或变量的地址才能进行调用,这个解析就是做了一部分这样的工作(说一部分的原因是因为多态的特性可能会导致在加载时不能确定具体的方法地址)
    • 初始化
      • 对静态变量进行赋值,执行静态方法块里的操作,这里的本质其实也是javac在编译时就收集好所有的静态操作,生成一个自动执行的静态方法
  • ClassLoader就是用来将class文件们正确加载到jvm的类,其实也就是实现了装载这个功能
  • 关于类加载和java类的本质只是非正式不严谨的讲述了一下,因为这篇文章如果再讲这些东西就太长了

Parent Delegate Model

其实这个模型就是一个委派(Delegate)模式

  • 当VM需要去加载类文件(字节码)时,会使用ClassLoader去把字节码读取到内存中,而每一个ClassLoader在需要load时,都会先尝试使用自己的父类Loader去加载,如果父类已经加载过了这个Class或者执行了加载操作,那子类就不必再去执行加载操作
  • 这样的好处是可以避免重复加载
  • 在JVM中,ClassLoader类的委托层次
    • Bootstrap ClassLoader: 最顶层的ClassLoader,负责加载JAVA_HOME/lib/下面的标准类库,即如果不是Java的标准类,那Bootstrap会忽略
    • Extension ClassLoader: 负责加载JAVA_HOME/lib/ext/下面所有的类
    • Application ClassLoader: 负责加载上述两种自带ClassLoader所不能加载的其他的CLASSPATH下的类,也就是说一般我们自己定义的类都是这个ClassLoader加载的
    • 注意这些类没有继承层次,都是通过delegate去进行调用的(也就是组合)
  • 在Android的Dalvik/ART中:
    • BootClassLoader:最顶层的ClassLoader,它相当于JVM中的BootstrapClassLoader,用来加载android的标准类库
    • BaseDexClassLoader:ClassLoader的子类,它是PathClassLoader和DexClassLoader的基类,它不是一个abstract类,它的constructor提供了几个参数
      • dexPath: 加载的apk或者jar的路径
      • optimizedDirectory:从apk或者jar/aar提取出来的odex文件存放的路径,null的话就会放在默认路径下
      • libraryPath:C/C++库的目录
      • parent:委托对象
    • PathClassLoader:optimizedDirectory设置为null,即提取出来odex存放在默认路径,系统加载我们自己定义的类就是用的这个ClassLoader,相当于ApplicationClassLoader,另外在art上貌似加载未安装的apk是可以的,但是在dalvik上是不可以的,这个classloader也不可以从网络上加载
    • DexClassLoader:optimizedDirectory是保留的,而且无论是在dalvik上还是art上都可以从任何地方加载odex文件,可以使用它加载我们指定路径上的odex文件,包括从网络上加载

How

  • defineClass是用来虚拟机用来将读入的字节流转化为Class对象的方法,即从磁盘上读出一个class/dex文件后,怎么样把它转化为对应的Class对象存储在对应的永生代/元空间内,这个方法是Java为我们写好的;在 Android上我们要下载源码才能看到这个方法,否则只是一个throw Exception

  • loadClass:这个方法定义了类加载的方式,Parent Delegate Model就是在loadClass里面制定的策略,它会先确认自己是否加载过这个类,然后再委托给parent,最后如果仍未加载,自己尝试加载(findClass);如果想要自己定义类加载策略,可以Override这个方法

  • findClass:定义了如何去寻找这个方法参数里指定的class,一般来说我们只需要重写这个方法

  • 定义好了如何使用:

    ClassLoader loader = new MyClassLoader(...);
    loader.loadClass("TestClass");
    

Why?

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

推荐阅读更多精彩内容

  • Android插件化基础的主要内容包括 Android插件化基础1-----加载SD上APKAndroid插件化基...
    隔壁老李头阅读 5,395评论 5 36
  • ClassLoader翻译过来就是类加载器,普通的java开发者其实用到的不多,但对于某些框架开发者来说却非常常见...
    时待吾阅读 1,055评论 0 1
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,523评论 18 399
  • JVM体系结构 JVM是一种解释执行class文件的规范技术。 我翻译的中文图: 类装载器子系统 在JVM中负责装...
    zhazhaxin阅读 11,687评论 7 69
  • 我的长期执行力一直很差,每年在准备考试的前几个月制定了学习计划,也往往没有坚持,来参加读写营就是为了养成每天读写的...
    萃_柳州阅读 224评论 0 0