【随笔】JVM核心:JVM运行和类加载

前言

本篇博客将写一点关于JVM的东西,涉及JVM运行时数据区、类加载的过程、类加载器、ClassLoader、双亲委派机制、自定义类加载器等,这些都是博主自己的一点理解,如果有误,欢迎大家评论拍砖~

关于JVM运行时数据区

JVM运行时数据区

关于类加载

class文件加载至内存,链接(校验、解析),初始化;最终形成JVM可以直接使用的JAVA类型的过程。

加载:在方法区形成类的运行时数据结构;在堆里面形成该类的Class对象,作为访问方法区的入口。

加载

链接:class文件是否存在问题;一些符号引号替换成直接引用。

初始化:初始化一个类,先初始化它的父类。虚拟机会保证一个类的初始化在多线程环境中被正确加锁和同步。

要使用类A,必须先加载类A;加载类A,就会把静态变量、静态块合并初始化,然后在调用构造器。注意类的加载和初始化,只有一次。

关于类加载器

上文已经说了,类加载器的作用就是:将class文件的字节码内容加载到内存中,并将这些静态数据转化成方法区中的运行时数据结构,在堆中生成一个代表这个类的Class对象,作为方法区类数据的访问入口。

类加载器的层次结构

引导类加载器bootstrap classloader 加载JAVA核心库($JAVA_HOME/jre/lib/rt.jar),原生代码实现(C++),并不继承自java.lang.ClassLoader。 扩展类加载器extensions classloader JAVA可以提供一个扩展目录($JAVA_HOME/jre/ext/*.jar)来加载Java类。 由sun.misc.Launcher.ExtClassLoader实现 应用程序类加载器application classloader(也称系统类加载器) 一般来说,JAVA应用的类由它加载,即加载路径是classpath下的路径。 由sun.misc.Launcher.AppClassLoader实现。 自定义类加载器 开发人员继承java.lang.ClassLoader实现自己的类加载器

类加载器的层次结构

关于java.lang.ClassLoader

ClassLoader的基本职责就是: 第一,根据指定的类名称,找到或者生成对应的字节码,并根据字节码生成class对象 第二,加载JAVA应用所需的资源,如配置文件等。

ClassLoader的组合模式

组合模式为双亲委派机制提供支持

demo:

类加载器的层次

引导类加载器是原生代码实现,我们获取不到,所以是null。

ClassLoader重要API

getParent():该类加载器的父类加载器 loadClass(String name):加载名称为name的类,并返回Class实例。 加载顺序是:先交给扩展类加载器加载,如果加载不到,交给引导类加载器加载,加载不到,交给自己去加载,如果自己也加载不到,那么ClassNotFoundException。【双亲委派机制】 如果要改变类的加载顺序,那么可以override该方法。 findClass(String name),不是加载,仅仅是查找而已 findLoadedClass(String name),查找已经被加载过的 defineClass(String name,byte[] b, int off ,int len),可以把字节数组的内容转换成JAVA类,并会返回Class实例。

类加载器的代理模式:双亲委派机制

类加载器的代理模式:就是把加载指定类的过程交给其他加载器。 JAVA默认使用的类加载器代理模式是:双亲委派机制。 双亲委派机制: 就是某个特定的类加载器接到加载类的请求时,首先将加载任务委托给父类加载器,依次追溯,比如说从应用加载器委托给扩展类加载器,从扩展类加载器委托给引导类加载器。这种委托,直至委托到层次最高的类加载器,即引导类加载器,如果委托的父类加载器可以完成加载任务,那么成功返回;只有父类加载器无法完成时,才去自己加载。 可以看出双亲委派机制的意思就是优先父类加载器加载! 试想如果我们定义了一个java.lang.String类,根据双亲委派机制,那么JDK只会加载它自己的String。这显然保证了Java核心库的类型安全。

双亲委派机制不是唯一的选择

虽然JDK默认的类加载机制是双亲委派机制,但是并不是所有都采用,比如有些服务器,如Tomcat,虽然也采用代理的方式加载,但是加载顺序却恰恰和双亲委派机制相反,它是首先尝试加载这个类,只有加载不到的情况下,才去让父类加载器代理加载。 为什么会这样呢,不是说双亲委派很安全么? 其实就是在安全,和灵活方面进行取舍!

写一个自定义类加载器

MyClassLoader:

自定义类加载器

重写findClass:

findClass

Test:

测试

一般情况下,自定义类加载器,需要继承自ClassLoader。 首先来说,可以检查请求的类是否已经被自定义的类加载器加载;如果加载了,那么直接返回;否则,那么交给父类加载器,就是进行双亲委派;如果双亲委派也加载不到,那么交给自定义类加载器进行“自定义的方式”来加载类。 另外,被2个不同的类加载加载的同一个类,JVM不会认为是一个类。

好了,关于JVM运行和类加载的过程就写到这里,^_^

2017.10.29 zhangfengzhe

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏企鹅号快讯

Python模块

Python模块 可以将代码量较大的程序分割成多个有组织的、彼此独立但又能相互交互的代码片段,这些自我包含的有组织的代码段就是模块 模块在物理形式上表现为以.p...

2507
来自专栏程序员的碎碎念

php常用函数(一)

parse_url:函数的作用是解析一个URL并返回一个包含其URL组件的关联数组,错误时返回false。例: ? 页面显示效果如下: ? ran...

3494
来自专栏CaiRui

Python之简单的用户登录和注册

#!/bin/bash/env python # -*- coding:utf-8 -*- def login(username,password): ...

35510
来自专栏闵开慧

mapreduce主程序如何传递变量到map或者reduce函数中使用

    一般我们写的mapreduce主程序放在客户端机器上,执行任务时是在集群机器上,所以要将变量从主程序传递到我们自己写的map或者reduce函数中就不能...

4215
来自专栏深度学习与计算机视觉

Python 新建文件夹与复制文件夹内所有内容

在指定路径下新建一个文件夹: import os def newfile(path): path=path.strip() path=path....

2276
来自专栏Python小屋

Python多线程编程中daemon属性的作用

在脚本运行过程中有一个主线程,若在主线程中创建了子线程,当主线程结束时根据子线程daemon属性值的不同可能会发生下面的两种情况之一: 如果某个子线程的daem...

3415
来自专栏阮一峰的网络日志

EOF是什么?

我学习C语言的时候,遇到的一个问题就是EOF。 它是end of file的缩写,表示"文字流"(stream)的结尾。这里的"文字流",可以是文件(file)...

3633
来自专栏数值分析与有限元编程

Fortran知识 | 代码错误(insufficient virtual memory)

如图所示,提示为:insufficient virtual memory ? 程序试图访问一个受保护或者不存在的地址。多数为可分配数组、指针等动态内存引发的错误...

3777
来自专栏性能与架构

nginx rewrite

rewrite的作用 rewrite可以实现url的重定向,把用户请求的url转发到另一个url,但用户浏览器地址并不改变 例如常用的伪静态化,就是通过rewr...

3685
来自专栏IT可乐

Linux系列教程(二十二)——Linux的bash变量

  上篇博客我们介绍了bash的一些基本功能,这是我们平时操作最频繁的。本篇博客我们介绍bash的变量,为后面编写shell脚本做铺垫。 1、什么是变量   变...

21710

扫码关注云+社区

领取腾讯云代金券