一文读懂 Java 反射机制那些事

简介:

前不久学习了反射机制,来总结下。在此之前,回顾下java程序的编译运行过程,分为三个阶段:源码(.java文件)进过编译生成字节码文件(.class文件),然后jvm加载字节码文件执行程序(runtime)。

前两个步骤(编译阶段)是在硬盘上完成的,后一个步骤(运行阶段)是在内存中完成的,而中间这个衔接就是:jvm通过类加载器----ClassLoader把硬盘中的class文件加载到内存中生成一个Class类的对象,这样就可以使用这个类中的成员变量和方法。一个类默认只会被加载一次,所以这个类对应的Class对象有且仅有一个。

什么是java反射机制?

1983年Smith首次提出反射这个概念,主要指程序可以访问、检测和修改他本身状态或行为的一种能力。

java反射机制是在运行状态中中对类进行解剖并操作类中的构造方法,成员方法,成员属性(主要用于框架中),这种动态获取信息以及动态调用对象的方法的功能称为java语言的反射机制。

Class对象和反射机制的联系。

了解了反射机制的概念,那么可见要想利用java反射机制做一些事,那么就要利用Class对象,所以说Class对象是反射的前提。

那么,怎么获取Class对象?

java中有三种方式获取Class对象:

 ●  类名.class
 ●  对象名.gerClass

 ●  Class.forName("全限定名(包名 + 类名)");

04e1ad8eb41464133579d5ae07413f87a538a07a

补充:Class对象分两种

1.普通Class对象:基于 引用类型

2.预定义(在jvm中的)Class对象:基于 基本类型 和 void

反射机制的几种作用:
 ●  在运行时判断任意一个对象所属的类
 ●  在运行时构造任意一个类的对象
 ●  在运行时判断任意一个类所具有的成员变量和方法
 ●  在运行时调用任意一个对象的方法

先准备一个类:


package com.test.demo;

public class Student {
public String name;
private int age;

public Student() {
}

private Student(String name, int age) {
this.name = name;
this.age = age;
}

public void show(String msg){
System.out.println("show方法 = " + msg);
}
private void speak(String msg,int number){
System.out.println("speak方法 = " + msg +":"+ number );
}

@Override
public String toString() {
return "Student{" + "name='" + name + '\'' + ", age=" + age +'}';
}
}
反射的使用1:构造器(Constructor)的反射

再次之前,我们可以通过公共的空参构造new一个Student,但是无法new私有的满参构造。

Student student = new Student();

现在来反射构造构造器(反射的形式创建实例)


public static void main(String[] args)
throws NoSuchMethodException, IllegalAccessException,
InvocationTargetException, InstantiationException {
//获取Class对象
Class<?> clazz = Student.class;
/*
根据参数类型获取相应的构造器
参数类型是形参类型
*/
Constructor<?> constructor = clazz.getConstructor();
/*
创建实例
参数类型是实参类型(形参一一对应)
*/
Object obj = constructor.newInstance();
System.out.println("obj = " + obj);
}

这样获取到的Student对象和new出来的空参构造器new出来的对象效果一样的(实际业务开发并没有意义)。

前者通过new创建出来对象的方式相比用反射创建的对象更被动,前者 是被new出来的,而用反射,是自己创建自己(对象),构造方法反客为主。

还有一种方式,就是直接通过Class对象创建构造器:


public static void main(String[] args)
throws IllegalAccessException, InstantiationException {
//获取Class对象
Class<?> clazz = Student.class;
/*
默认调用空参构造创建一个实例
jdk9中已过时
*/
Object obj = clazz.newInstance();
System.out.println("obj = " + obj);
}

在Student类中 ,还有一个私有的构造器,正常方式下是不能通过私有构造器创建对象的。,但是反射可以做到:


public static void main(String[] args)
throws NoSuchMethodException, IllegalAccessException,
InvocationTargetException, InstantiationException {
//获取Class对象
Class<?> clazz = Student.class;
/*
获取构造
因为权限是私有,但getConstructor()只能获取public修饰的方法
getDeclaredConstructor():获取声明的方法。只要声明的就可以
*/
Constructor<?> constructor = clazz.getDeclaredConstructor(String.class, int.class);
System.out.println("满参私有构造 :" + constructor);
/*
私有构造,newInstance会产生非法访问异常:java.lang.IllegalAccessException
所以要改变权限setAccessible() -->暴力反射
*/
constructor.setAccessible(true);
Object obj = constructor.newInstance("小明",20);

System.out.println("obj = " + obj);
}

以上就是利用反射来创建一个对象(反射构造器)。

反射的使用2:方法(Method)的反射

接下来看看Student对象内两个方法的反射

我们之前(外部)使用方法,都是都是通过对象调用(非私有)方法,如果是静态方法就是类直接调用。

那么,使用反射调用(非私有)方法,该怎么做?


public static void main(String[] args)
throws NoSuchMethodException, IllegalAccessException,
InvocationTargetException {
//获取Class对象
Student student = new Student();
Class<? extends Student> clazz = student.getClass();
/*
getMethod():获取Class对象里的方法
参数一:方法名
参数二:参数列表类型
*/
Method show = clazz.getMethod("show", String.class);
/*
调用show方法需要对象和参数
invoke()方法:调用的意思
参数一:调用此方法的对象
参数二:调用此方法需要传入的实参
*/
show.invoke(student, "hello public show");
}

反射可以理解为语言语法上的倒装句:

我们平时写代码都是我(对象)去调用方法,这里就是:

new Student().show("对象调用方法");

而在 show.invoke(student, "hello public show"); 中,

show方法考虑的是谁来调用我,然后Student对象说,我来调用你(student作为参数)。

扩展:如果公共的show方法加上static关键字,会影响方法调用吗?

提示:静态与对象无关.

答:加上static关键字,普通代码即使不new对象也可以调用,这个大家都知道,那么,在show.invoke(student, "hello public show"); 中参数1 写 null 也是不影响的,因为,show方法来自于 Student的Class对象。

接下来看看私有方法的反射如何实现?

ps: 反射通道的API都很有规律,可读性很强


public static void main(String[] args)
throws NoSuchMethodException, IllegalAccessException,
InvocationTargetException {
//获取Class对象
Student student = new Student();
Class<? extends Student> clazz = student.getClass();
/*
getDeclaredMethod():获取Class对象里的声明过的方法(包括)
参数一:方法名
参数二:参数列表类型
*/
Method speak = clazz.getDeclaredMethod("speak", String.class, int.class);
//私有方法,暴力反射
speak.setAccessible(true);
/*
调用show方法需要对象和参数
invoke()方法:调用的意思
参数一:调用此方法的对象
参数二:调用此方法需要传入的实参
*/
speak.invoke(student, "hello private speak",2018);
}
反射的使用3:属性(Field)的反射

在Student实体中有一个共有属性一个私有属性,我们可以通过对象来设置共有属性的值,那么通过反射如何实现所有属性的赋值?

先来看看共有属性name的赋值


public static void main(String[] args)
throws ClassNotFoundException, NoSuchFieldException,
IllegalAccessException, InstantiationException {
//获取Class对象,参数全限定名
Class<?> clazz = Class.forName("com.test.demo.Student");
/*
getField():通过属性名获取属性
*/
Field name = clazz.getField("name");
//获取对象
Object obj = clazz.newInstance();
/*
设置一个值
参数一:哪个对象的属性值
参数二:参数
*/
name.set(obj,"张三");
System.out.println(obj);
}

根据前面说的API,反射属性不难理解。

私有属性的反射也不难实现


public static void main(String[] args)
throws ClassNotFoundException, NoSuchFieldException,
IllegalAccessException, InstantiationException {
//获取Class对象,参数全限定名
Class<?> clazz = Class.forName("com.test.demo.Student");
/*
getDeclaredField():通过属性名获取(所有权限)属性
*/
Field age = clazz.getDeclaredField("age");
//暴力反射
age.setAccessible(true);
//创建对象
Object obj = clazz.newInstance();
/*
设置一个值
参数一:哪个对象的属性值
参数二:参数
*/
age.set(obj,20);
System.out.println(obj);
}
总结:

使用java的反射机制,一般需要遵循三步:

 ●  获得你想操作类的Class对象
 ●  通过第一步获得的Class对象去取得操作类的方法或是属性名
 ●  操作第二步取得的方法或是属性

那么反射到底有什么用?

反射最主要还是运用在框架中,了解反射才更好的了解一些框架的原理。


原文发布时间为:2018-11-21

本文作者:四夕又欠

本文来自云栖社区合作伙伴“ Web项目聚集地”,了解相关信息可以关注“Web项目聚集地”。

相关文章
|
10月前
|
设计模式 人工智能 安全
AQS:Java 中悲观锁的底层实现机制
AQS(AbstractQueuedSynchronizer)是Java并发包中实现同步组件的基础工具,支持锁(如ReentrantLock、ReadWriteLock)和线程同步工具类(如CountDownLatch、Semaphore)等。Doug Lea设计AQS旨在抽象基础同步操作,简化同步组件构建。 使用AQS需实现`tryAcquire(int arg)`和`tryRelease(int arg)`方法以获取和释放资源,共享模式还需实现`tryAcquireShared(int arg)`和`tryReleaseShared(int arg)`。
500 32
AQS:Java 中悲观锁的底层实现机制
|
8月前
|
人工智能 缓存 安全
Java中的反射机制:深入探索与应用
Java反射机制是程序运行时动态获取类信息并操作类成员的特性,具备高度灵活性,但也伴随性能与安全风险。本文详解反射的基本用法、高级应用及最佳实践,助你掌握这一强大工具的正确使用方式。
198 0
|
10月前
|
人工智能 Java 关系型数据库
Java——SPI机制详解
SPI(Service Provider Interface)是JDK内置的服务提供发现机制,主要用于框架扩展和组件替换。通过在`META-INF/services/`目录下定义接口实现类文件,Java程序可利用`ServiceLoader`动态加载服务实现。SPI核心思想是解耦,允许不同厂商为同一接口提供多种实现,如`java.sql.Driver`的MySQL与PostgreSQL实现。然而,SPI存在缺陷:需遍历所有实现并实例化,可能造成资源浪费;获取实现类方式不够灵活;多线程使用时存在安全问题。尽管如此,SPI仍是Java生态系统中实现插件化和模块化设计的重要工具。
478 0
|
8月前
|
人工智能 前端开发 安全
Java开发不可不知的秘密:类加载器实现机制
类加载器是Java中负责动态加载类到JVM的组件,理解其工作原理对开发复杂应用至关重要。本文详解类加载过程、双亲委派模型及常见类加载器,并介绍自定义类加载器的实现与应用场景。
344 4
|
8月前
|
人工智能 安全 Java
掌握Java反射:在项目中高效应用反射机制
Java反射是一种强大功能,允许程序在运行时动态获取类信息、创建对象、调用方法和访问字段,提升程序灵活性。它在框架开发、动态代理、注解处理等场景中广泛应用,如Spring和Hibernate。但反射也存在性能开销、安全风险和代码复杂性,应谨慎使用。
177 0
|
9月前
|
人工智能 Java
Java中的反射机制:深入探索与应用
本文介绍了Java反射机制的基本概念、用途及其实现方式。反射机制允许程序在运行时动态获取类的属性和方法,并调用它们,适用于处理私有成员或权限受限的情况。文章详细讲解了`Class`类的功能,包括获取类的方法、属性、注解、构造器等信息,以及通过四种方式获取`Class`对象的示例代码。此外,还探讨了类加载器、继承关系判断、动态代理等高级内容,展示了如何在运行时创建接口实例并处理方法调用。文末提供了完整的代码示例以加深理解。
205 0
Java中的反射机制:深入探索与应用
|
10月前
|
Java 区块链 网络架构
酷阿鲸森林农场:Java 区块链系统中的 P2P 区块同步与节点自动加入机制
本文介绍了基于 Java 的去中心化区块链电商系统设计与实现,重点探讨了 P2P 网络在酷阿鲸森林农场项目中的应用。通过节点自动发现、区块广播同步及链校验功能,系统实现了无需中心服务器的点对点网络架构。文章详细解析了核心代码逻辑,包括 P2P 服务端监听、客户端广播新区块及节点列表自动获取等环节,并提出了消息签名验证、WebSocket 替代 Socket 等优化方向。该系统不仅适用于农业电商,还可扩展至教育、物流等领域,构建可信数据链条。
|
缓存 Dubbo Java
理解的Java中SPI机制
本文深入解析了JDK提供的Java SPI(Service Provider Interface)机制,这是一种基于接口编程、策略模式与配置文件组合实现的动态加载机制,核心在于解耦。文章通过具体示例介绍了SPI的使用方法,包括定义接口、创建配置文件及加载实现类的过程,并分析了其原理与优缺点。SPI适用于框架扩展或替换场景,如JDBC驱动加载、SLF4J日志实现等,但存在加载效率低和线程安全问题。
609 7
理解的Java中SPI机制
|
10月前
|
人工智能 JavaScript Java
Java反射机制及原理
本文介绍了Java反射机制的基本概念、使用方法及其原理。反射在实际项目中比代理更常用,掌握它可以提升编程能力并理解框架设计原理。文章详细讲解了获取Class对象的四种方式:对象.getClass()、类.class、Class.forName()和类加载器.loadClass(),并分析了Class.forName()与ClassLoader的区别。此外,还探讨了通过Class对象进行实例化、获取方法和字段等操作的具体实现。最后从JVM类加载机制角度解析了Class对象的本质及其与类和实例的关系,帮助读者深入理解Java反射的工作原理。
259 0
|
11月前
|
存储 Java 编译器
Java 中 .length 的使用方法:深入理解 Java 数据结构中的长度获取机制
本文深入解析了 Java 中 `.length` 的使用方法及其在不同数据结构中的应用。对于数组,通过 `.length` 属性获取元素数量;字符串则使用 `.length()` 方法计算字符数;集合类如 `ArrayList` 采用 `.size()` 方法统计元素个数。此外,基本数据类型和包装类不支持长度属性。掌握这些区别,有助于开发者避免常见错误,提升代码质量。
1001 1