直接上结论:
1、类名.class , 不执行静态代码块和构造器
2、对象.getClass() , 执行静态代码块和构造器
3、Class.forName() , 只执行静态代码块,不执行构造器
先定义一个person类
public class Person {
public Person() {
System.out.println("构造器执行啦!");
}
// 可以做资源的初始化
static {
System.out.println("静态代码块执行啦!");
}
}
1、类名.class
public static void main(String[] args) throws ClassNotFoundException {
Class<Person> personClass = Person.class;
Class<Person> personClass2 = Person.class;
//Person p = new Person();
//p.getClass();
//p.getClass();
//Class.forName("com.test.demo.Person");
//Class.forName("com.test.demo.Person");
}
运行结果:
类名.class:
1、JVM将使用类装载器, 将类装入内存(前提是:类还没有装入内存),不做类的初始化工作.返回Class的对象
2、如果类已经装入内存就不会再重复装载
3、不会运行静态代码块和构造器
2、对象.getClass()
public static void main(String[] args) throws ClassNotFoundException {
//Class<Person> personClass = Person.class;
//Class<Person> personClass2 = Person.class;
Person p = new Person();
p.getClass();
p.getClass();
//Class.forName("com.test.demo.Person");
//Class.forName("com.test.demo.Person");
}
运行结果:
对象.getClass():
1、运行静态代码块和构造器(实例化对象的时候就执行了)
3、Class.forName()
public static void main(String[] args) throws ClassNotFoundException {
//Class<Person> personClass = Person.class;
//Class<Person> personClass2 = Person.class;
//Person p = new Person();
//p.getClass();
//p.getClass();
Class.forName("com.test.demo.Person");
Class.forName("com.test.demo.Person");
}
运行结果
Class.forName():
1、只执行静态代码块,且只执行一次
4、创建对象的过程
首先要给大家区分一个观点,加载与类加载是两个截然不同的过程。Java的“类加载”是一个类从被加载到虚拟机内存中开始,到卸载出虚拟机内存为止的整个生命周期中的一个过程,包括加载,验证,准备,解析,初始化五个阶段。
而“加载”指的是类加载的第一个阶段,加载阶段,虚拟机需要完成以下3件事情:
1.通过一个类的全限定名来获取定义此类的二进制字节流2.将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构
3.在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据结构的访问入口。
验证,确认类型符合Java语言的语义,检查各个类之间的二进制兼容性(比如final的类不用拥有子类等),另外还需要进行符号引用的验证。
准备,Java虚拟机为类变量分配内存,设置默认初始值。
解析(可选的),在类型的常量池中寻找类,接口,字段和方法的符号引用,把这些符号引用替换成直接引用的过程。
类中的静态块会在整个类加载过程中的初始化阶段执行,而不是在类加载过程中的加载阶段执行。初始化阶段是类加载过程中的最后一个阶段,该阶段就是执行类构造器方法的过程,方法由编译器自动收集类中所有类变量(静态变量)的赋值动作和静态语句块中的语句合并生成,一个类一旦进入初始化阶段,必然会执行静态语句块。所以说,静态块一定会在类加载过程中被执行,但不会在加载阶段被执行。
来源: