面试:说说Java反射中获取Class对象三种方式的区别?

简介: 面试:说说Java反射中获取Class对象三种方式的区别?

在上篇文章《一篇文章全面了解Java反射机制》中我们学习了Java反射机制的基本使用,留心的朋友可能已经注意到了,在文中提到了三种获取Class对象的方法。

如果面试中涉及到Java反射,那么遇到该面试题的概率将大大增加。

以下三种获取Class对象的方式有什么不同?

1、new Object().getClass 2、Object.class 3、 Class.forName("java.util.String")

本篇文章就通过实例带大家来了解一下这三种获取Class对象的区别。示例基于JDK8。

实例演示场景一

为了更好的演示,我们先创建一个对象Person,对象内部定义了一些静态的方法。















public class Person {
  static {    System.out.println("Person:静态代码块");  }
  {    System.out.println("Person:动态代码块");  }
  public Person(){    System.out.println("Person:构造方法");  }}

在Person对象中分别定义了:静态代码块、动态代码块、构造方法。

针对上面的实例,我们构建了三个单元测试的场景,对应代码如下:


















public class GetClassTest {
  @Test  public void test1(){    Class<?> clz = Person.class;  }
  @Test  public void test2() throws ClassNotFoundException {    Class<?> clz = Class.forName("com.choupangxia.reflect.Person");  }
  @Test  public void test3() {    Class<?> clz = new Person().getClass();  }}

分别执行三个单元测试发现,第一个单元测试没打印任何内容;第二个单元测试打印了“静态方法”中的内容;第三个单元测试打印出了全部内容:

Person:静态代码块
Person:动态代码块
Person:构造方法

也就是说通过Person.class的方法获取对象的Class对象,根本不会调用对象中任何的代码块或代码。而Class.forName()会调用静态代码块的内容。

而第三种方式打印所有内容的原因很显然,就因为要先实例化对象。

实例演示场景二

下面再组合一下这三种方式,看看一些其他的效果。

首先,依次调用三个获取Class对象的方法:









@Testpublic void test4() throws ClassNotFoundException {  Class<?> clz = Person.class;  System.out.println("---------------");  clz = Class.forName("com.choupangxia.reflect.Person");  System.out.println("---------------");  clz = new Person().getClass();}

test4打印日志如下:

---------------
Person:静态代码块
---------------
Person:动态代码块
Person:构造方法

通过日志说明,Class.forName()方法执行过静态代码块之后,new Person().getClass()就不再会执行同样的静态代码块了。这也证明静态代码块只会被初始化一次。

再调整组合第二种场景:







@Testpublic void test5() throws ClassNotFoundException {  Class<?> clz = new Person().getClass();  System.out.println("---------------");  clz = Class.forName("com.choupangxia.reflect.Person");}

test5打印日志如下:

Person:静态代码块
Person:动态代码块
Person:构造方法
---------------

同样只打印一次静态代码块的操作。再次证明类中的静态代码块只会被初始化一次。

实例演示场景三

这里,我们比较一下三种形式获得的Class对象是否是相同的。测试代码如下:










@Testpublic void test6() throws ClassNotFoundException {  Class<?> clz1 = Person.class;  Class<?> clz2 = Class.forName("com.choupangxia.reflect.Person");  Class<?> clz3 = new Person().getClass();
  System.out.println(clz1 == clz2);  System.out.println(clz2 == clz3);}

注意,上面我们使用的是等号,也就是说比较的是引用。猜猜打印的结果?

true
true

三种形式获得的Class对象是同一个对象。这是为什么呢?

这要涉及到类的加载过程,我们知道类加载过程分:加载阶段、连接阶段和初始化阶段。

类的加载阶段是将class文件中的二进制数据读取到内存中,然后将该字节流所代表的静态存储结构转化为方法区中运行时的数据结构,并且在堆内存中生成一个该类的java.lang.class对象,作为方法区数据结构的入口。

类加载阶段的最终产物是堆内存中的class对象,对于同一个Classloader对象,不管某类被加载多少次,对应堆内存中的class对象始终只有一个。

也就是说无论通过哪种形式来获取Class对象,获得的都是堆内存中对应的Class对象。

回顾三种形式

(1)类名.class:JVM将使用类装载器,将类装入内存(前提是:类还没有装入内存),不做类的初始化工作,返回Class的对象。

(2)Class.forName("类名字符串"):装入类,并做类的静态初始化,返回Class的对象。

(3)实例对象.getClass():对类进行静态初始化、非静态初始化;返回引用运行时真正所指的对象(子对象的引用会赋给父对象的引用变量中)所属的类的Class的对象。


目录
相关文章
|
1月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
76 2
|
21小时前
|
Java 数据库连接 Spring
反射-----浅解析(Java)
在java中,我们可以通过反射机制,知道任何一个类的成员变量(成员属性)和成员方法,也可以堆任何一个对象,调用这个对象的任何属性和方法,更进一步我们还可以修改部分信息和。
|
26天前
|
Java 程序员
Java社招面试题:& 和 && 的区别,HR的套路险些让我翻车!
小米,29岁程序员,分享了一次面试经历,详细解析了Java中&和&&的区别及应用场景,展示了扎实的基础知识和良好的应变能力,最终成功获得Offer。
66 14
|
1月前
|
存储 缓存 算法
面试官:单核 CPU 支持 Java 多线程吗?为什么?被问懵了!
本文介绍了多线程环境下的几个关键概念,包括时间片、超线程、上下文切换及其影响因素,以及线程调度的两种方式——抢占式调度和协同式调度。文章还讨论了减少上下文切换次数以提高多线程程序效率的方法,如无锁并发编程、使用CAS算法等,并提出了合理的线程数量配置策略,以平衡CPU利用率和线程切换开销。
面试官:单核 CPU 支持 Java 多线程吗?为什么?被问懵了!
|
1月前
|
存储 算法 Java
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
本文详解自旋锁的概念、优缺点、使用场景及Java实现。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
|
21天前
|
Java
java中面向过程和面向对象区别?
java中面向过程和面向对象区别?
21 1
|
1月前
|
存储 缓存 安全
java 中操作字符串都有哪些类,它们之间有什么区别
Java中操作字符串的类主要有String、StringBuilder和StringBuffer。String是不可变的,每次操作都会生成新对象;StringBuilder和StringBuffer都是可变的,但StringBuilder是非线程安全的,而StringBuffer是线程安全的,因此性能略低。
49 8
|
1月前
|
Java 编译器 程序员
Java面试高频题:用最优解法算出2乘以8!
本文探讨了面试中一个看似简单的数学问题——如何高效计算2×8。从直接使用乘法、位运算优化、编译器优化、加法实现到大整数场景下的处理,全面解析了不同方法的原理和适用场景,帮助读者深入理解计算效率优化的重要性。
35 6
|
1月前
|
监控 Java
Java基础——反射
本文介绍了Java反射机制的基本概念和使用方法,包括`Class`类的使用、动态加载类、获取方法和成员变量信息、方法反射操作、以及通过反射了解集合泛型的本质。同时,文章还探讨了动态代理的概念及其应用,通过实例展示了如何利用动态代理实现面向切面编程(AOP),例如为方法执行添加性能监控。
|
2月前
|
IDE Java 编译器
java的反射与注解
java的反射与注解
23 0