Java反射详解

本文详细介绍了Java反射机制,从反射概述、为何需要反射到反射的实现方式,包括Class类的理解、获取Class对象、创建运行时类的对象、访问和调用成员变量与方法。反射在框架设计中扮演重要角色,允许程序在运行时动态加载和操作类的内部信息,提升代码的灵活性和可扩展性。
摘要由CSDN通过智能技术生成

1. 反射概述

  • 反射(Reflection)被视为动态语言的关键,反射机制允许程序在运行时借助于Reflection API获取任何类的内部信息,并能直接操作任意对象的内部属性及方法
  • 框架 = 反射 + 注解 + 设计模式
  • 反射是框架设计的灵魂

2.为什么需要反射?

2.1创建对象的过程

要理解为什么需要反射,首先看一下运行A a=new A();时发生了什么:

  • 编写好的代码会编译成一个.class文件(字节码文件),然后被类加载器加载进JVM内存中的方法区中,并创建了类A的Class对象到堆中【不是new出来的对象,而是类的类型对象,同一个类只会有一个Class对象,作为方法区类的数据结构的接口】。
  • jvm创建对象前,会先检查该类是否加载,寻找类对应的Class对象,若已加载,则为对象分配内存进行初始化,也就是代码:new A()。
  • 以上的过程可以理解为“正”
    创建
  • 上面的情况下对象是我们自己new的,程序相当于写死了让JVM去跑。但是实际应用中服务器会突然遇到某个请求要用到某个类,但是还没加载进JVM,难道要停下来自己写段代码再new一下然后再启动服务器?显然这是不可能的,因此就需要另想办法。
  • 因此就需要程序在运行时,能够根据需要动态的加载一些类。例如数据库有时是用mysql,有时用oracle,就需要动态地根据实际情况加载驱动类,这个时候反射就有用了,程序可以写得比较动态化,通过Class clz = Class.forName("驱动类的全类名");通过类的全类名(传入的参数)让JVM在服务器中找到并加载这个类。这就体现出了反射的好处了,也即Java语言的动态性!
  • 使用反射的好处:
    • 在程序运行过程中操作对象
    • 可以解耦,提高程序的可扩展性

3. 什么是反射?

  • 反射机制允许在程序运行时获得任何类的内部信息[属性、方法、构造器等],并能直接调用任意对象的属性和方法,这种动态获取类的信息以及动态调用对象的方法的功能称为java的反射机制。
  • 反射把类中的各个组成部分(属性、方法、构造器等)分别映射成一个个Java对象
    • 一个类有:成员变量、方法、构造器、父类和接口等信息,利用反射技术可以对一个类进行解剖,把各种成分分别映射成一个个对象
    • (其实:类中这些成员方法、构造方法,在加入类中都有一个类来描述)
  • 反射的本质:Class对象的由来是将class文件读入内存,并为之创建一个Class对象,然后反向获取对应类的信息

4. Class类的理解

  • 程序经过编译(javac.exe命令)以后,会生成一个或多个字节码文件(.class结尾)。接着我们使用java.exe命令对某个字节码文件进行解释运行。相当于将某个字节码文件加载到JVM内存中,此过程就称为类的加载。加载到内存中的类,我们就称为运行时类,此运行时类,就作为Class的一个实例。
  • Class和类的关系可以类比类和对象的关系,Class可以理解为是所有其他类的类,每个加载到内存中的类对应一个Class的实例,Class类的实例表示正在运行的 Java 程序中的类和接口(包括基本数据类型)
  • 加载到内存中的运行时类,会缓存一定的时间。在此时间之内,我们可以通过不同的方式来获取此运行时类。

5. 反射的使用

  • Class、Field、Method、Constructor四个对象的关系:
    在这里插入图片描述

5.1 获取Class对象

  • 在反射中,要获取一个类或调用一个类的方法,我们首先需要获取到该类的 Class 对象,获取Class对象有三种方法:

1.使用 Class.forName() 静态方法

Class clz = Class.forName("java.lang.String");
  • 将字节码文件加载进内存,返回Class对象。
  • 当知道该类的全路径名时,可使用该方法。 多用于配置文件,将类名定义在配置文件中。读取文件,加载类

2. 使用类名.class 属性

Class clz = String.class;
  • 这种方法只适合在编译前就知道操作的 Class。多用于参数的传递

3. 使用obj.getClass() 方法

String str = new String("Hello");
Class clz = str.getClass();
  • getclass()方法在object类中定义着
  • 多用于对象的获取字节码的方式

5.2 创建运行时类的对象

  • 通过反射创建类对象主要有两种方式:通过Class对象的newInstance()方法、通过Constructor 对象的newInstance()方法。
  • newInstance()内部调用了运行时类的空参的构造器,因此必须保证运行时类有无参构造器

1. 通过Class对象的 newInstance() 方法。

Class clz = Person.class;
Person p1 = (Person)clz.newInstance();

2. 通过Constructor对象的 newInstance() 方法

Class clz = Person.class;
Constructor constructor = clz.getConstructor();
Person p2 = (Person)constructor.newInstance();
  • 通过 Constructor 对象创建类对象可以选择特定构造方法,而通过 Class 对象则只能使用默认的无参数构造方法。下面的代码就调用了一个有参数的构造器进行了类对象的初始化。
Class clz = Person.class;
Constructor constructor = clz.getConstructor(String.class, int.class);//参数姓名和年龄
Person p3 = (Person)constructor.newInstance("张三", 25);

5.3 获取成员变量并调用

  • Field[] getFields():获取所有public修饰的成员变量(包括父类)
  • Field[] getDeclaredFields():获取所有的成员变量(不包括父类)
  • Field getField(String name):获取指定名称的public修饰的成员变量
  • Field getDeclaredField(string name):获取指定名称的成员变量
  • 设置成员变量的值:void set(Object obj,Object value):

    参数说明:1. obj:要设置的字段所在的对象;2. value:要为字段设置的值;

  • 获取值:get(object obj)

Student类:

package fanshe.field;
 
public class Student {
   
	public Student(){
   
		
	}
	//**********字段*************//
	public String name;
	protected int age;
	char sex;
	private String phoneNum;
	
	@Override
	public String toString() {
   
		return "Student [name=" +
  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值