类的加载 连接与初始化
加载:查找并加载类的二进制数据
连接
- 验证:确保被加载的类的正确性
- 准备:为类的静态变量分配内存,并将其初始化为默认值
- 解析:把类中的符号引用转换成为直接引用
初始化:为类的静态变量赋予正确的初始值
主动使用
创建类的实例
访问某个类或接口的静态变量,或者对该静态变量赋值
调用类的静态方法
反射(如class.forname(“”))
初始化一个类的子类
java虚拟机启动时被标明为启动类的类
Jdk1.7开始提供的动态语言支持:
Java.lang.invoke.MethodHandle实例的解析结果REF_invokeStatic,REF_putStatic,Ref_INVOKEsTATIC句柄对应的类没有初始化则初始化
类的加载
类的加载是指将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在内存中创建一个JAVA.LANG.CLASS对象(规范并未说明class对象位于哪个位置,hotspot虚拟机将其放在方法区中)用来封装类在方法区内的数据结构
加载.class文件的方式
- 从本地系统中直接加载
- 通过网络下载.class文件
- 通过.zip,jar等归档文件中加载.class文件
- 从专有的数据库中提取.class文件
- 将java源文件动态编译成.class文件(动态代理 因为类是在运行期间产生的 web开发时的jsp也是如此 )
jvm指令
-xx:+
-xx:-
-xx:
--xx:+TraceClassLoading 用于追踪类的加载信息 并打印出来
public class mytest{
public static void main(String [] arg){
System.out.println(p.str);
}
}
class p{
public static final String str="hellorold";
static{
System.out.println("static block")
}
}
//这里的输出结果不会输出static块里面的内容
//是因为 final代表常量 所以不能别改变 所以在编译阶段这个常量会被存储到调用这个常量的方法所在的类的常量池中
//本质上调用类并没有直接引用到定义常量的类,因此并不会触发定义常量的类的初始化
//注意:这里指的是将常量存储于mytest的常量池中,之后mytest与p就没有任何关系
//甚至我们可以将p的class文件删除
助记符
- ldc表示将int,float或者是String类型的常量值从常量池中推送至栈顶
- bipush表示将单字节(-128~127)的常量值推送至栈顶
- sipush表示将一个短整型常量值(-32768~32767)推送至栈顶
- iconst_1表示将int类型的1推送至栈顶(由m1最多到5)
- anewarray表示创建一个引用类型的(如类接口数组)的数组,并将其引用至压入栈顶
- newarray:表示创建一个指定的原始类型(如int float char等)的数组,并将其引用值压入栈顶
public class test{
public static void main(String [] arg){
System.out.println(p.str);
}
}
class p{
public static final String str=UUID.randonUUID().toString();
static{
System.out.println("static block");
}
}
//这里会输出static块内容 因为当一个常量的值并非编译期间可以确定的,name其值就不会被放到调用类的常量池中,这是在程序运行时,会导致主动使用这个常量所在的类,显然会导致这个类被初始化
数组
对于数组实例来说,其类型是由jvm在运行期动态生成的,表示【lcom.xiguaapp.jvm.classloader.test这种形式。动态生成的类型,其父类型就是object
对于数组来说,javadoc经常将构成数组的元素为component,实际上就是将数组降低一个维度后的类型。
类的初始化时机
当java虚拟机初始化一个类时,要求它的所有父类都已经被初始化,但是这条规则并不适用于接口。
- 在初始化一个类时,并不会先初始化它所实现的接口。
- 在初始化一个接口时,并不会先初始化它的父接口。
因此,一个父接口并不会因为它的子接口或者实现类的初始化而初始化。只有当程序首次使用特定接口的静态变量时,才会导致该接口初始化。
类加载器双亲委托机制
在父亲委托机制中,各个加载器按照父子关系形成了树形结构,除了根类加载器之外,其余的类加载器都有且只有一个父加载器
比如,如果用自定义加载器loader1加载自定义class,在加载的时候,由loader1先判断是否有父类加载器,如果有那么父类加载器在先判断是否有父类加载器,如果没有则执行,如果不能执行那么则往子层,子层如果不可以那么再找子层的子层。通常都是由系统类加载器进行加载类。
类加载器加载的路径
Bootstrap classLoader -> Load Jre\lib\rt.jar或者是-Xbootclasspath选项指定的jar包
Extension classLoader ->Load jre\lib\ext*.jar或者-Djava.ext.dirs指定目录下的jar包
App classLoader ->load ClassPath或者-Djava.class.path所指定的目录下的类和jar包
Custom classLoader ->通过java.lang.classLoader的子类自定义加载class