java类加载

类加载机制
虚拟机把class文件加载到内存,并对数据进行校验,转换解析和初始化,形成可以被虚拟机直接使用的java类型,即java.lang.Class

1. 装载(Load)

查找和导入class文件

a) 通过一个类的全限定名获取定义次类的二进制字节流

b) 将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构

c) 在java堆中生成一个代表这个类的java.lang.Class对象,作为对方法区中这些数据的访问入口。

2. 链接(Link)

2.1 验证(Verify)

保证被加载类的正确性

  • 文件格式验证
  • 元数据验证
  • 字节码验证
  • 符号引用验证

2.2 准备(Prepare)

为类的静态变量分配内存,并将其初始化为默认值

2.3 解析(Resolve)

把类中的符号引用转换为直接引用

3.初始化


对类的静态变量,静态代码块执行初始化操作

类装载器ClassLoader

在装载(Load)阶段,其中第(1)步:通过类的全限定名获取其定义的二进制字节流,需要借助类装载器完成,顾名思义,就是用来装载Class文件的。

1.分类

  • Bootstrap ClassLoad 负责加载$JAVA_HOME中jre/lib/rt.jar里所有的class或Xbootclasspath选项指定的jar包。由C++实现,不是ClassLoader子类
  • Extension ClassLoader负责加载java平台中扩展功能的一些jar包,包括$JAVA_HOME中jre/lib/*.jar或-Djava.ext.dirs指定目录下的jar包。
  • App ClassLoader负责加载classpath中指定的jar包及Djava.class.path所指定目录下的类和jar包。
  • Custom ClassLoader通过java.lang.ClassLoader的子类自定义加载class,属于应用程序根据自身需要定义的ClassLoader,如tomcat、jboss都会根据j2ee规范自行实现ClassLoader.

2.加载原则

检查某个类是否已经加载:顺序是自底向上,从Custom ClassLoader到BootStrap ClassLoader逐层检查,只要某个ClassLoader已加载,就视为已加载此类,保证此类只被所有ClassLoader加载一次。

加载顺序:加载的顺序是自顶向下,也就是由上层来逐层尝试加载此类。

双亲委派机制:

定义:如果一个类加载器在接到加载类的请求时,它首先不会自己尝试去加载这个类,而是把这个请求任务委托给父类加载器完成,依次递归,如果父类加载器可以完成此类加载任务,就成功返回;只有父类加载器无法完成此加载任务时,才自己去加载。

优势:Java类随着加载它的类加载器一起聚类一种优先级的层次关系。比如:java中的Object类,它存放在rt.jar之中,无论哪一个类加载器需要加载这个类,最终都是委派给处于模型最顶端的启动类加载器进行加载,因此Object在各种类加载环境中都是同一个类。如果不采用双亲委派模型,那么由各个类加载器自己去加载的话,那么系统中会存在多种不同的Object类。

破坏双亲委派模型

线程上下文类加载器

双亲委派模型并不能解决Java应用开发中会遇到的类加载器的全部问题。

Java提供了很多服务提供者接口(SPI),允许第三方为这些接口提供实现。

常见的SPI有JDBC,JCE,JNDI,JAXP和JBI等。这些SPI的接口由Java核心库提供,如JAXP的SPI接口定义包含在javax.xml.parses中。

这些SPI的实现代码很可能是作为Java应用所依赖的jar包被包含进来,可以通过类路径(ClassPath)来找到,如实现了JAXP SPI的Apache Xerces锁包含的jar包。SPI接口中的代码经常需要加载具体的实现类。如JAXP中的javax.xml.parses.DocumentBuilderFactory类中的newInstance()方法用来生成一个新的DocumentBuilderFactory的实例。

这里的实例的真正的类是继承自java.xml.parsers.DocumentBuilderFactory,由SPI的实现锁提供的。

而问题在于,SPI的接口是Java核心库的一部分,由Bootstrap类加载器加载的,而SPI实现的Java类一般是由系统类加载器加载的。引导类加载器是无法找到SPI的实现类的,因为它只加载Java的核心库。

它也不能委派给系统类加载器,因为它是系统类加载器的祖先类加载器。也就是说,类加载器的双亲委派模型无法解决这个问题。

为了解决这个问题,Java设计团队只好引入了一个不太优雅的设计:线程上下文类加载器(Thread Context ClassLoader)

有了线程上下文类加载器,就可以做一些“舞弊”的事情了,JNDI服务使用这个线程上下文类加载器去加载所需要的SPI代码,也就是父类加载器请求子类加载器去完成类加载器的动作,这种行为实际上就是打通了双亲委派模型的层次结构来逆向使用类加载器,已经违背了双亲委派模型的一般性原则。

JDBC为什么要破坏双亲委派模型

因为类加载器收到加载范围的限制,在某些情况下父类加载器无法加载到需要的文件,这时候就需要委托子类加载器去加载class文件。

JDBC的Driver接口定义在JDK中,其实现由各个数据库的服务商来提供,比如MySQL驱动包。DriverManager类中要加载各个实现了Driver接口的类,然后进行管理,但是DriverManager位于$JAVA_HOME中jre/lib/rt.java包,由BootStrap类加载器加载,而其Driver接口的实现类是位于服务商提供的jar包,

根据类加载机制,当被状态的类引用了另外一个类的时候,虚拟机就会使用装载第一个类的类装载器装载被引用的类。

这就是说BootStrap类加载器还要去加载jar包中的Driver接口的实现类。
我们知道,BootStrap类加载器默认只负责加载$JAVA_HOME中的jre/lib/rt.jar里的所有的class,所以需要由子类加载器去加载Driver实现,这就破坏了双亲委派模型。

Tomcat为什么要破坏双亲委派模型

每个Tomcat的webappClassLoader加载自己的目录下的class文件,不会传递给父类加载器

事实上,tomcat之所以造了一堆自己的classLoader,大致是出于下面三个目的:

  • 对于各个webapp中的classlib,需要相互隔离,不能出现一个应用加载的类库会影响另一个应用的情况,而对于许多应用,需要共享的lib以便不浪费资源。

  • jvm一样的安全性问题。使用单独的classloader去装载tomcat自身的类库,以免其他恶意或无意的破坏。

  • 热部署。

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