Java开发岗位面试被问到反射怎么办
作者:蛋挞学姐 发布时间:2023-01-07 00:09:12
到底什么是反射呢???
反射的核心就是JVM在运行时才动态加载类或调用方法,访问属性,它不需要事先(写代码的时候或编译期)知道运行对象是谁。
每一个类都会产生一个对应的Class对象,也就是保存在.class文件。
所有类都是在对其第一次使用时,动态加载到JVM的,当程序创建一个对类的静态成员的引用时,就会加载这个类,Class对象仅在需要的时候才会加载,static初始化是在类加载时进行的。
public class TestMain {
public static void main(String[] args) {
System.out.println(Test.name); // 对Test类的静态成员name引用。
}
}
class Test {
public static String name = "Test Name";
static {
System.out.println("Test静态块");
}
public Test() {
System.out.println("Test构造了");
}
}
输出:
Test静态块 Test Name
2. 类的生命周期
一个类编译完成后,下一步就是开始使用类,怎么使用?
类编译完成后,开始使用类,在程序执行中JVM通过装载,链接,初始化这3个步骤完成。
1.装载:由类加载器完成,找到对应的字节码,创建一个Class对象。
类加载器首先会检查这个类的Class对象是否已经被加载过,如果没有加载,默认的类加载器就会根据类名查找对应的.class文件。
加载器将.class文件的二进制文件装入JVM的方法区,并且在堆区创建描述这个类的java.lang.Class对象,用来封装数据,但是同一个类只会被类装载器装载一次。
2.链接:就是把二进制数据组装为可以运行的状态
校验:一般用来确认此二进制文件是否适合当前的JVM(版本)
准备:为静态成员分配内存空间,并设置默认值。
解析:转换常量池中的代码作为直接引用的过程,直到所有的符号都可以被运行程序使用(建立完整的对应关系)验证类中的字节码,为静态域分配空间。
3.初始化:如果该类有父类,则对其初始化,执行静态初始化器和静态初始化块。
3. Java反射框架主要提供以下功能:
在运行时构造任意一个类的对象
在运行时调用任意一个对象的方法
在运行时判断任意一个对象所属的类
在运行时判断任意一个类所具有的成员变量和方法(通过反射甚至可以调用private方法)
反射的基本用法
1. 获得Class对象
<1> 使用Class类的forName静态方法:
public static Class<?> forName(String className)
在JDBC开发中常用此方法加载数据库驱动: Class.forName(driver);
//加入Java开发交流君样:756584822一起吹水聊天
<2> 直接获取某一个对象的class:(编译时已知类型名称或已知对象)
Class<?> klass = int.class;
Class<?> classInt = Integer.TYPE;
<3> 调用某个对象的getClass()方法:
StringBuilder str = new StringBuilder("123");
Class<?> klass = str.getClass();
注意: 使用.class来创建Class对象的引用时,不会自动初始化该Class对象,使用forName(…)会自动初始化该Class对象。
2. 判断是否为某个类的实类
一般:用instanceof关键字判断
反射:反射中Class对象的isInstance()方法
public native boolean isInstance(Object obj);
3.创建实例
通过反射来生成对象主要有两种方式:
<1> 使用Class对象的newInstance()方法来创建Class对象对应类的实例。
Class<?> c = String.class;
Object str = c.newInstance();
<2> 先通过Class对象获取指定的Constructor对象,再调用Constructor对象的newInstance()方法来创建实例。
// 获取String所对应的Class对象
Class<?> c = String.class;
// 获取String类带一个String参数的构造器
Constructor constructor = c.getConstructor(String.class);
// 根据构造器创建实例
Object obj = constructor.newInstance("23333");
System.out.println(obj);
4. 获取构造器信息
主要是通过Class类的getConstructor方法得到Constructor类的一个实例,而Constructor类有一个newInstance方法可以创建一个对象实例
5. 获取方法
<1> getDeclaredMethods()
– 返回类或接口声明的所有方法,包括公共、保护、默认(包)访问和私有方法,但不包括继承的方法
public Method[] getDeclaredMethods() throws SecurityException
<2> getMethods()
– 返回某个类的所有公共(public)方法,包括继承的公有方法
public Method[] getMethods() throws SecurityException
<3> getDeclaredMethod()
– 返回一个特定的方法,其中第一个参数为方法名称,后面的参数为方法的参数类型对应的Class对象
public Method getDeclaredMethod(String name, Class<?>... parameterTypes)
<4> getMethod()
– 返回一个特定的方法,其中第一个参数为方法名称,后面的参数为方法的参数类型对应的Class对象
public Method getMethod(String name, Class<?>... parameterTypes)
6. 获取类的成员变量(字段)信息
getDeclaredFields()
– 访问所有已声明的成员变量,但不能访问继承的成员变量。
getFileds()
– 访问所有已声明的公有(public)成员变量,包括继承的公有成员变量。
getDeclaredField()
– 特定访问所有成员变量(不包括继承的),参数为成员变量的名字。
getFiled()
– 特定访问公有成员变量(包括继承的),参数为成员变量的名字。
7. 利用反射创建数组
Class<?> cls = Class.forName("java.lang.String");
Object array = Array.newInstance(cls, 25);
//往数组里添加内容
Array.set(array, 0, "hello");
Array.set(array, 1, "Java");
Array.set(array, 2, "fuck");
Array.set(array, 3, "Scala");
Array.set(array, 4, "Clojure");
//获取某一项的内容
System.out.println(Array.get(array,3));
//加入Java开发交流君样:756584822一起吹水聊天
其中的Array类为java.lang.reflect.Array
类,我们通过Array.newInstance()
创建数组对象,它的原型是:
public static Object newInstance(Class<?> componentType, int length) throws NegativeArraySizeException {
return newArray(componentType, length);
}
newArray()方法是一个Native方法:
private static native Object newArray(Class<?> componentType, int length) throws NegativeArraySizeException;
反射的注意事项
由于反射会额外消耗一定的系统资源,因此如果不需要动态地创建一个对象,那么就不需要用反射。反射调用方法时可以忽略权限检查,因此可能会破坏封装性而导致安全问题。
反射的主要用途
最重要的用途就是开发各种通用框架
很多框架(比如Spring)都是配置化的(比如通过XML文件配置JavaBean,Action之类的),为了保证框架的通用性,它们可能需要根据配置文件加载不同的对象或类,调用不同的方法,这个时候就必须用到反射——运行时动态加载需要加载的对象。
来源:https://blog.csdn.net/wj1314250/article/details/117264941


猜你喜欢
- Java 存储模型和共享对象详解很多程序员对一个共享变量初始化要注意可见性和安全发布(安全地构建一个对象,并其他线程能正确访问)等问题不是很
- 在Android开发中,有时我们需要对SQLite数据库进行增,删,查,找等操作,现在就来简单介绍一下,以下为详细代码。 &nbs
- 自从SEOTcs系统11月份24日更新了一下SEO得分算法以来,一直困扰我的一个问题出现了,java的数据job任务,在执行过程中会经常报以
- 1) async / await使用 async / await 模式,可以在执行代码块操作的时候不会阻塞 UI 或者当前的线程。即使该操作
- MyBatis是一个支持普通SQL查询,存储过程和高级映射的优秀持久层框架。MyBatis消除了几乎所有的JDBC代码和参数的手工设置以及对
- 本文基于SpringBoot 2.5.0-M2讲解Spring中Lifecycle和SmartLifecycle的作用和区别,以及如何控制S
- sqlite是啥?1、一种轻型数据库2、关系型数据库3、占用资源很低,几百K内存,适合嵌入式设备4、支持windows、linux、unix
- MyEclipse配置IDEA配置Tomcat环境IDEA:2020.2Tomcat:apache-tomcat-9.0.38创建Web项目
- maven项目install时忽略执行test在项目所在文件夹根目录使用maven命令打包时<!-- 不执行单元测试,也不编译测试类
- 问题发现今天发生了一件事,令我非常郁闷,就是我在使用一个SDK时,当我调用他的方法时,提示我方法中的参数var1, var2如下:// 方法
- 一、NIO基本简介NIO (New lO)也有人称之为java non-blocking lO是从Java 1.4版本开始引入的一个新的IO
- 本文实例讲述了Java编程实现提取文章中关键字的方法。分享给大家供大家参考,具体如下:实现代码:/** * 相关的jar包 * lucene
- 前言 SpringCloud 是微服务中的翘楚,最佳的落地方案。 在微服务架构中多层服务之间会相互调用,如果其中有一
- 对于本地图片我们可以通过selector来轻松的实现点击态。 但是在我们的项目中,一个关于对非本地图片的点击态实现还是难倒了不少人;因此专门
- 概述从今天开始, 小白我将带大家开启 Java 数据结构 & 算法的新篇章.链表链表 (Linked List) 是一种递归的动态数
- 这篇文章主要介绍了Spring Batch批处理框架使用解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需
- 面试题:1同步方法和同步块,哪种更好?2.如果同步块内的线程抛出异常会发生什么?1. 同步方法和同步块,哪种更好?同步块更好,这意味着同步块
- 实践过程效果代码class BaseClass{ public class Win32
- C#实现委托namespace Delegate{ delegate void DGSayiHi(string n
- 为什么需要全局异常处理在传统 Spring Boot 应用中, 我们 @ControllerAdvice 来处理全局的异常,进行统一包装返回