自由的羽毛 发表于 2024-8-26 06:37:59

【JVM】类加载器、双亲委派、SPI(二)

类加载器、双亲委派、SPI

类加载器加载的类如何存储

https://i-blog.csdnimg.cn/direct/3475961154bc4e2593d047e6822e5b78.png


[*]1.Class是访问类型T界说的Java程序入口,在Java代码中,如果你想获取T的界说,如查看其方法界说,字段界说,首先要获取到对应的Class实例
[*]2.ClassLoader实例本身对Java而言,也是一个分配在堆中的一个对象,它管理着自己在方法区的一个区域
元空间内部给各个类加载器划分了内存区域。另外还需要留意一点的是,方法区会有碎片化问题,方法区的垃圾回收通常被称为类的卸载。方法区回收之后是没有整理的
双亲委派机制

https://i-blog.csdnimg.cn/direct/470784983fee4c1f92af393825abd8b5.png
如果一个类加载器收到了加载某个类的请求,则该类加载器并不会去加载该类,而是把这个请求委派给父类加载器,每一个条理的类加载器都是云云,因此全部的类加载器请求终极都会传送到顶端的启动类加载器;只有当父类加载器在其搜刮范围内无法找到所需的类,并将该结果反馈给子类加载器,子类加载器会尝试自己加载
如何打破双亲委派?为什么要打破双亲委派机制?

由于在某些情况下父类加载器需要委托子类加载器去加载class文件。受到加载范围的限制,父类加载器无法加载到需要的文件,以Driver接口为例,由于Driver接口界说在jdk当中的,而其实现由各个数据库的服务商来提供,比如mysql的就写了MySQL Connector,那么问题就来了,DriverManager(也由jdk提供)要加载各个实现了DriverManager接口的实现类,然后进行管理,但是DriverManager由启动类加载器加载,只能加载JAVA_HOME的lib目录下的文件,而其实现是由服务商提供的,由系统类加载器加载,这个时间就需要启动类加载器来委托子类来加载Driver实现,从而破坏了双亲委派机制。类似这样的情况就需要打破双亲委派。打破双亲委派的意思其实就是不委派、向下委派。
线程上下文类加载器

https://i-blog.csdnimg.cn/direct/99dcbb181b9948b4a78ca6fd0baec092.png
1.是什么?一种特殊的类加载器,可以通过Thread获取,基于此可实现逆向委托加载
2.为什么?为了解决双亲委派的缺陷而生
3.怎么做?如图所示
SPI机制
它是一种服务发现机制。它通过在ClassPath路径下的META-INF/services文件夹查找文件,主动加载文件里所界说的类。这一机制为很多框架扩展提供了大概,比如在Dubbo、JDBC中都利用到了SPI机制。Tomcat/Spring就是这样类似的机制
沙箱安全

https://i-blog.csdnimg.cn/direct/39d2bab1f6264d7ab92243cf29f5e669.png
比如界说了一个类名为String地点包为java.lang,由于这个类本身是属于jdk的,如果没有沙箱安全机制的话,这个类将会污染到我全部的String,但是由于沙箱安全机制,以是就委托顶层的bootstrap加载器查找这个类,如果没有的话就委托extension,extension没有就到appclassloader,但是由于String就是jdk的源代码,地点bootstrap那边就加载到了,先找到先利用,以是就利用bootstrap里面的String后面的一概不能利用,这就包管了不被恶意代码污染
https://i-blog.csdnimg.cn/direct/0753863f0554420abe3b841a88e432ca.png
openjdk源码会有很多这样的判断AccessController.doPrivileged
反射的底层原理

https://i-blog.csdnimg.cn/direct/77c22c328bd1400285871a1b4bdc642a.png
涉及到的字节码指令


[*]forName
[*]getField
[*]getMethod
[*]https://i-blog.csdnimg.cn/direct/041ef1c3997840fca4bd2d11c37ce658.png
通过这些方法获取一个类的信息,那么它是怎么存储的呢?它会去方法区中进行查找,那么在HotSpot中它是通过Dictionary字典来存储这些信息的,底层数据结构是hashtable
key:类的全限定名+类加载器->index
value: Metadata:klass
反射时需要先找到InstsanceKlass对象然后才气找到InstanceMirrorKlass对象,由于JVM是没法找到堆中的Class对象的,JVM找到InstanceKlass,就可以直接拿到InstanceMirrorKlass

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
页: [1]
查看完整版本: 【JVM】类加载器、双亲委派、SPI(二)