Java反射总结(史上最全,有这一篇就够了)

我的视频课:《Java反射精讲》欢迎大家围观~

一、概述

什么是反射?

Java反射机制指的是在Java程序运行状态中,对于任何一个类,都可以获得这个类的所有属性和方法;对于给定的一个对象,都能够调用它的任意一个属性和方法。

这种动态获取类的内容以及动态调用对象的方法称为反射机制

为什么使用反射?

在计算机科学领域,反射是指一类能够自我描述和自控制的应用。

在Java编程语言中,反射是一种强有力的工具,是面向抽象编程的一种实现方式,它能使代码语句更加灵活,极大提高代码的运行时装配能力。Java反射机制允许编程人员在对类未知的情况下,获取类相关信息的方式变得更加多样灵活,调用类中相应方法,是Java增加其灵活性与动态性的一种机制。

总结一下,Java反射机制有如下作用

  1. 反射机制极大的提高了程序的灵活性和扩展性,降低模块的耦合性,提高自身的适应能力;
  2. 通过反射机制可以让程序创建和控制任何类的对象,无需提前硬编码目标类;
  3. 使用反射机制能够在运行时构造一个类的对象、判断一个类所具有的成员变量和方法、调用一个对象的方法;
  4. 反射机制是构建框架技术的基础所在,使用反射可以避免将代码写死在框架中。

二、反射的原理

我们知道了什么是反射以及反射的作用,那么在Java中是如何支持反射的呢?

首先我们需要了解Java程序运行的过程,该过程包含两个阶段:编译运行

在程序编译阶段,Java代码会通过JDK编译成 .class字节码文件;
在程序运行阶段,JVM会去调用业务逻辑对应需要的的字节码文件,生成对应的Class对象,并调用其中的属性方法完成业务逻辑。

Java的反射机制原理就是在程序运行阶段,主动让JVM去加载某个 .class文件生成Class对象,并调用其中的方法和属性。 如下图:

Java类加载过程
注:

  • Class类在java.lang包中,继承了Object;
  • Class对象的由来是将.class文件读入内存,并为之创建一个Class对象,一个.class文件对应一个Class对象;

三、反射的使用

通过第二节的原理描述,我们知道使用Java反射时,有一个息息相关的类——Class类,其实不单单是Class类,还有三个主要使用的类,如下表:
在这里插入图片描述
下面我们来一一介绍:

1、获取Java类(Class类的使用)

这是使用Java反射的第一步,获取了对应的Class类之后,我们就可以生成对象,然后调用对象的方法和属性,这一步有三种实现方式。

方法一

        Book book = new Book();
		Class bookClass = book.getClass();
		
		//输出类名
		System.out.println(bookClass.getName());

这个方法其实是Object的一个方法,Class继承了Object,所以我们可以直接使用。

方法二

        Class bookClass = Book.class;
		
		//输出类名
		System.out.println(bookClass.getName());

方法三

	    Class bookClass;
		try {
			bookClass = Class.forName("test.Book");
			//输出类名
			System.out.println(bookClass.getName());
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		}

方法三是使用最多的方法。

2、获取类的构造函数(Constructor类的使用)

我们获取到一个类的Class对象之后,可以调用Class对象的getDeclaredConstructors()方法获取该类的构造函数,如下:

	
	    // 反射所有声明的构造方法
	public static void reflectAllConstructor() {
		System.out.println(TAG + "=============获取所有的声明的构造函数==============");
		try {
			Class<?> classBook = Class.forName("test.Book");
			Constructor<?>[] constructorsBook = classBook
					.getDeclaredConstructors();
			for (Constructor constructor : constructorsBook) {
				System.out.println(TAG + constructor);
			}
		} catch (Exception ex) {
			ex.printStackTrace();
		}
	}

获取了构造函数之后,调用Constructor类对象的newInstance()即可构造出我们想要类的对象,如下:

Book book = (Book)constructor.newInstance();

3、获取类的函数(Method类的使用)

当我们得到了一个Class对象之后,我们可以获取该类的所有方法,如下:

    // 反射所有的public的函数
	public static void reflectPublicMethods() {
		System.out.println(TAG + "=============获取所有的public的函数==============");
		try {
			Class<?> classBook = Class.forName("test.Book");
			Method[] methodsBook = classBook.getMethods();
			for (Method method : methodsBook) {
				System.out.println(TAG + method);
			}

		} catch (Exception ex) {
			ex.printStackTrace();
		}
	}

	// 反射所有的声明的方法
	public static void reflectAllMethods() {
		System.out.println(TAG + "=============获取所有的声明的函数==============");
		try {
			Class<?> classBook = Class.forName("test.Book");
			Method[] methodsBook = classBook.getDeclaredMethods();
			for (Method method : methodsBook) {
				System.out.println(TAG + method);
			}

		} catch (Exception ex) {
			ex.printStackTrace();
		}
	}

getDeclaredMethods()和getMethods()都可以获取到类的方法,那么他们有什么区别呢?

getMethods()执行结果如下:
在这里插入图片描述
我们看到 getMethods()不仅只获取了自己定义的公用方法(private获取不了),还把Object父类的公用方法也获取了

getDeclaredMethods()执行结果如下:
在这里插入图片描述
我们看到 getDeclaredMethods()只能获取自己类中定义的方法,但是可以获取到private方法

4、获取类的属性(Field类的使用)

当我们得到了一个Class对象之后,我们可以获取该类的所有属性,代码如下:

// 反射所有的public的属性
	public static void reflectPublicFields() {
		System.out.println(TAG + "=============获取所有的public的属性==============");
		try {
			Class<?> classBook = Class.forName("test.Book");
			Field[] fieldsBook = classBook.getFields();
			for (Field field : fieldsBook) {
				System.out.println(TAG + field);
			}

		} catch (Exception ex) {
			ex.printStackTrace();
		}
	}

	// 反射所有的声明的属性
	public static void reflectAllFields() {
		System.out.println(TAG + "=============获取所有的声明的属性==============");
		try {
			Class<?> classBook = Class.forName("test.Book");
			Field[] fieldsBook = classBook.getDeclaredFields();
			for (Field field : fieldsBook) {
				System.out.println(TAG + field);
			}

		} catch (Exception ex) {
			ex.printStackTrace();
		}
	}

同Methods,获取属性也有getDeclaredFields()和getFields()两种。

四、反射带来的问题

反射虽然能够给我带来诸多便利,但是反射有一个致命问题:反射的效率比直接调用低很多,所以我们要慎用反射。

至于为什么反射效率低?

其实很好理解,使用反射时要通知JVM加载.class文件并且生成Class对象,而直接调用则不用,所以反射效率低。







我的视频课

下面是我录制的一些视频课,欢迎大家围观~

《Android性能优化参考》

本课程包含了Android中的App启动优化、UI优化、内存优化、图片优化、耗电量等常见的性能优化场景,通过学习此课程,你将对整个Android性能优化体系有清晰的认识。

性能优化作为Android高级开发的必备技能,也是大厂面试必考的题目,是体现一个人技术深度最好的试金石。

《面试之排序算法》

排序算法是我们面试被问到最多的基础算法,本课程详细介绍了七种排序算法,包括插入排序、选择排序、冒泡排序、谢尔排序、快速排序、堆积排序和二路并归排序。每种算法都详细介绍了核心思想、详细步骤、时间复杂度和代码实现,希望帮助大家深入理解排序算法,搞定面试!

《Android混合App开发实战》

本课程为Android HyBrid App开发实战课程,由浅入深,从三种App的历史和特点开始,介绍了Android WebView的使用、Java和JS交互的原生方式、著名的WebView安全漏洞、JSBridge的原理和使用,最后通过一个网上商城的实战综合全部内容,让同学们掌握并深入理解Android HyBrid App开发。

《AI导论》

介绍人工智能AI的诞生历史和到现在为止的不同发展阶段;介绍了AI领域中常见的名词概念和其关系,包括机器学习、深度学习、神经网络结构搜索 NAS、生成对抗网络 GAN等;最后对AI发展做出展望。
本课程属于导论课程,旨在帮助同学们从宏观层面把握AI,建立AI的知识体系。

《Java注解精讲》

本课程详细介绍了Java中的注解机制,包括注解的定义和分类,注解的使用和自定义,注解的源码和架构分析;
本课程语言简单凝练,视频短小精悍,让你一次彻底搞懂Java注解!

《Java反射精讲》

反射是Java中重要的也是较难理解的内容;
本课程从反射的定义、作用、原理和使用出发,全方位帮你彻底搞定反射;

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

不死鸟JGC

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值