标签:父类 数组 优先级 操作 初始 lang test ack 序列化
加载-验证-准备-解析-初始化-使用-卸载
把.class二进制数据读到内存中,并放到方法区,然后在堆中创建一个Java.lang.Class对象,这个对象就是用来封装类在方法区的数据结构的。
所以,类加载机制的最终产物是:在堆中创建了java.lang.Class对象,这个对象提供了访问方法区内部数据结构的接口。
这个主要就是验证包的签名等
为静态变量分配内存,并设置初始值。
内存分配动作发生在方法区的,在准备阶段,给类成员进行初始化。
类型 | 初始化值 |
---|---|
String | null |
Object | null |
int | 0 |
将符号引用转成直接引用
说白了,就是,将变量换成内存中真实地址,都将高级的东西解析为机器识别的底层东西。
初始化有生命类变量和静态代码块2种方式,它们的优先级相同,谁在前面谁先来。
static int age = 20;
static{
System.out.println("hello,world");
System.out.println(age);
}
package com.siyu;
public class Claloader {
static int age = 20;
static{
System.out.println("hello,world");
System.out.println(age);
}
public static void getName(){
String name = Thread.currentThread().getName();
System.out.println(name);
}
}
package com.siyu;
public class Test {
public static void main(String[] args) {
// 调用类的某个静态方法,触发类的初始化
Claloader.getName();
// hello,world
// 20
// main
}
}
package com.siyu;
public class Claloader {
static int age = 20;
static{
System.out.println("hello,world");
System.out.println(age);
}
public static void getName(){
String name = Thread.currentThread().getName();
System.out.println(name);
}
}
package com.siyu;
public class Test {
public static void main(String[] args) {
// 调用类的某个静态方法,触发类的初始化
int age = Claloader.age;
// hello,world
// 20
}
}
package com.siyu;
public class Claloader {
static int age = 20;
static{
System.out.println("hello,world");
System.out.println(age);
}
public static void getName(){
String name = Thread.currentThread().getName();
System.out.println(name);
}
}
class Sub extends Claloader{
static String name = "nezha";
static {
System.out.println("这是子类的静态代码块");
}
}
package com.siyu;
public class Test {
public static void main(String[] args) {
int age = Sub.age;
// hello,world
// 20
}
}
JVM加载字节码文件靠的是类加载器,这个操作是在JVM外部实现的。
这样应用程序就可以自己决定如何获取所需的类。
如果两个类来自同一个Class文件,但是由不同的类加载器加载,那么者两个类一定是不相等。
从JVM角度讲,只有两种加载器。一种是启动类加载器,是虚拟机自身的一部分,由C++语言实现;
还有就是其他类加载器,由Java语言实现,全都继承自抽象类java.lang.ClassLoader 独立于虚拟机外部。
从开发角度看,主要分为这三种:
启动类加载器(Bootstrap ClassLoader):加载<JAVA_HOME>\lib目录中,或者被 -Xbootclasspath参数所指定的路径中。
扩展类加载器(Extension ClassLoader):主要加载<JAVA_HOME>\lib\ext目录中的
应用程序类加载器(Application ClassLoader):加载用户类路径(ClassPath)上所指定的类库。如果我们没有自定义过自己的类加载器,那么这就是程序默认的类加载器。
在使用类加载器加载类的过程种,最好遵循双亲委派模型。双亲委派的原理是:类加载器收到加载类的请求时,先把这个请求委派给父类加载去完成,每一层次的加载器按这个这个逻辑执行。那么所有的加载请求最终都应该传送到顶层的启动类加载中。父加载器无法加载,子加载器才会自己加载。这样做的好处是可以避免类的重复加载,保证程序运行的稳定性。
标签:父类 数组 优先级 操作 初始 lang test ack 序列化
原文地址:https://www.cnblogs.com/hellosiyu/p/13236090.html