JavaSE学习笔记25-反射(reflection)

news/2025/2/23 16:39:43

反射

在Java中,反射(Reflection) 是一种强大的机制,允许程序在运行时检查和操作类、方法、字段等信息。通过反射,可以动态地创建对象、调用方法、访问字段,甚至修改私有成员。反射的核心类是 java.lang.reflect 包中的 ClassMethodFieldConstructor 等。

反射的常见用法

以下是反射的常见用法

1. 获取 Class 对象

Class 对象是反射的核心,表示一个类或接口的元数据。可以通过以下方式获取 Class 对象:

java">// 1. 通过类名获取
Class<?> clazz = Class.forName("java.lang.String");

// 2. 通过对象获取
String str = "Hello";
Class<?> clazz = str.getClass();

// 3. 通过类字面量获取
Class<?> clazz = String.class;

2. 创建对象

通过反射可以动态创建类的实例:

java">Class<?> clazz = Class.forName("java.lang.String");
// 使用无参构造函数创建对象
Object obj = clazz.getDeclaredConstructor().newInstance();

 如果构造函数有参数,可以这样:

java">Constructor<?> constructor = clazz.getDeclaredConstructor(String.class);
Object obj = constructor.newInstance("Hello, Reflection!");

3. 调用方法

通过反射可以动态调用方法:

java">Class<?> clazz = Class.forName("java.lang.String");
Object obj = clazz.getDeclaredConstructor().newInstance();

// 获取方法对象
Method method = clazz.getMethod("substring", int.class, int.class);

// 调用方法
String result = (String) method.invoke(obj, 0, 5);
System.out.println(result); // 输出子字符串

4. 访问字段

通过反射可以访问和修改字段的值:

java">class MyClass {
    private String name = "Reflection";
}

Class<?> clazz = MyClass.class;
Object obj = clazz.getDeclaredConstructor().newInstance();

// 获取字段对象
Field field = clazz.getDeclaredField("name");

// 设置字段可访问(即使是私有字段)
field.setAccessible(true);

// 获取字段值
String nameValue = (String) field.get(obj);
System.out.println(nameValue); // 输出: Reflection

// 修改字段值
field.set(obj, "New Value");
System.out.println(field.get(obj)); // 输出: New Value

5. 访问私有成员

通过 setAccessible(true) 可以绕过访问控制,访问私有成员(字段、方法、构造函数):

java">class MyClass {
    private void privateMethod() {
        System.out.println("Private Method Called!");
    }
}

Class<?> clazz = MyClass.class;
Object obj = clazz.getDeclaredConstructor().newInstance();

// 获取私有方法
Method method = clazz.getDeclaredMethod("privateMethod");

// 设置方法可访问
method.setAccessible(true);

// 调用私有方法
method.invoke(obj); // 输出: Private Method Called!

反射的优缺点

优点:

  • 动态性:可以在运行时动态加载类、调用方法、访问字段。

  • 灵活性:适用于框架和库的开发(如 Spring、Hibernate)。

缺点:

  • 性能开销:反射操作比直接调用慢。

  • 安全性问题:可以绕过访问控制,破坏封装性。

  • 代码可读性差:反射代码通常难以理解和维护。


反射的使用场景

  • 框架开发:如 Spring 的依赖注入、Hibernate 的 ORM 映射。

  • 动态代理:如 JDK 动态代理。

  • 工具类:如 JSON 库(Gson、Jackson)的序列化和反序列化。


反射示例代码

以下是一个完整的反射示例:

java">import java.lang.reflect.*;

public class ReflectionExample {
    public static void main(String[] args) throws Exception {
        // 获取 Class 对象
        Class<?> clazz = Class.forName("java.lang.String");

        // 创建对象
        Object obj = clazz.getDeclaredConstructor(String.class).newInstance("Hello, Reflection!");

        // 调用方法
        Method method = clazz.getMethod("substring", int.class, int.class);
        String result = (String) method.invoke(obj, 0, 5);
        System.out.println(result); // 输出: Hello

        // 访问字段(假设有一个私有字段)
        Field field = clazz.getDeclaredField("value");
        field.setAccessible(true);
        byte[] value = (byte[]) field.get(obj);
        System.out.println(new String(value)); // 输出: Hello, Reflection!
    }
}

 通过反射,Java 提供了强大的动态能力,但应谨慎使用,避免滥用导致性能问题和代码复杂性。

静态vs动态语言


动态语言
    是一类在运行时可以改变其结构的语言:例如新的函数、对象、甚至代码可以被引进,
    已有的函数可以被删除或是其他结构上的变化。通俗说就是在运行时代码可以根据某些条件改变自身结构
    主要动态语言:Object-C、C#、Javascript、PHP、Python等。
静态语言
    与动态语言相对应的,运行时结构不可变的语言就是静态语言。如Java、C、C++
    Java不是动态语言,但Java可以称之为“准动态语言”。即Java有一定的动态性,我们可以利用反射机制获得
    类似动态语言的特性。Java的动态性让编程的时候更加灵活!

笔记

Reflection(反射)是Java被视为动态语言的关键,反射机制允许程序在执行期借助于Reflection API取得任何类的内部信息,
并能直接操作任意对象的内部属性及方法
    语法: Class c = Class.forName("java.lang.String")
加载完类之后,在堆内存的方法区中就产生了一个Class类型的对象(一个类只有一个Class对象),这个对象就
包含了完整的类的结构信息。我们可以通过这个对象看到类的结构。这个对象就像一面镜子,透过这个镜子看到类的结构,
所以,我们形象的称之为:反射
    正常方式:引入需要的“包类”名称 --》 通过new实例化 --》取得实例化对象
    反射方式:实例化对象 --》getClass()方法 --》 得到完整的“包类”名称

Java反射机制提供的功能
    在运行时判断任意一个对象所属的类
    在运行时构造任意一个类的对象
    在运行时判断任意一个类所具有的成员变量和方法
    在运行时获取泛型信息
    在运行时调用任意一个对象的成员变量和方法
    在运行时处理注解
    生成动态代理
    ....

优点:实现动态创建对象和编译,体现出很大的灵活性

缺点:对性能有影响。使用反射基本上是一直解释操作,我们可以告诉JVM,我们希望做什么并且它满足我们的要求。
这类操作总是慢于 直接执行相同的操作

反射相关的主要API
java.lang.Class:代表一个类
java.lang.reflect.Method:代表类的方法
java.lang.reflect.Filed:代表类的成员变量
java.lang.reflect.Constructor:代表类的构造器
...

Class类


在Object类中定义了以下的方法,此方法将被所有子类继承
    public final Class getClass()
    这个方法的返回值的类型是一个Class类,此类是Java反射的源头
    实际上所谓反射从程序的运行结果看也很好理解,即:可以通过对象反射求出类的名称

对象照镜子后可以得到的信息,某个类的属性,方法,和构造器,某个类到底实现了哪些接口
对于每个类而言,JRE都为其保留一个不变的Class类型的对象,一个Class对象包含了特定某个
结构(class/interface/enum/annotation/primitive type/void/[])的有关信息

Class本身也是一个类
Class对象只能由系统建立对象
一个加载的类在JVM中只会有一个Class实例
一个Class对象对应的是一个加载到JVM中的一个.class文件
每个类的实例都会记得自己是由哪个Class实例所生成
通过Class可以完整地得到一个类中的所有被加载的结构
Class类是Reflection的根源,针对任何你想动态加载,运行的类,唯有先获得相应的Class对象

Class类的常用方法
static ClassforName(String name)  返回指定类名name的Class对象
Object newInstance()              调用缺省构造函数,返回Class对象的一个实例
getName()                         返回此Class对象所表示的实体(类,接口,数组类或void)的名称
Class getSuperClass()             返回当前Class对象的父类的Class对象
Class[] getinterface()            获取当前Class对象的接口
ClassLoader getClassLoader()      返回该类的类加载器
Constructor[] getConstructors()   返回一个包含某些Constructor对象的数组
Method getMethod(String name,Class..T)  返回一个Method对象,此对象的形参类型为paramType
Field[] getDeclaredFields()       返回Field对象的一个数组

 


http://www.niftyadmin.cn/n/5863588.html

相关文章

Flutter开发的应用页面非常多时如何高效管理路由

文章目录 1. 集中式路由管理示例&#xff1a; 2. 动态路由生成 (onGenerateRoute)示例&#xff1a; 3. 模块化路由管理示例&#xff1a; 4. 使用路由管理库使用go_router的示例&#xff1a; 5. 路由分层管理总结 当Flutter应用中有大量页面时&#xff0c;路由管理变得复杂。为了…

第二届粤港澳大湾区数字经济与人工智能国际学术会议(DEAI 2025)

重要信息 2025年3月28-30日 I 广东省东莞市(广东科技学院-松山湖校区&#xff09; I www.icdeai.com 简介 第二届粤港澳大湾区数字经济与人工智能(DEAI 2025)将在2025年3月28-30日在广东省东莞市隆重举行。来自国内外高等院校、科学研究所、企事业单位的专家、教授、学者、…

Python 高级特性-切片

目录 切片 练习 小结 掌握了Python的数据类型、语句和函数&#xff0c;基本上就可以编写出很多有用的程序了。 比如构造一个1, 3, 5, 7, ..., 99的列表&#xff0c;可以通过循环实现&#xff1a; L [] n 1 while n < 99:L.append(n)n n 2 取list的前一半的元素&am…

常用高压缩率的视频容器格式,并进行大比例压缩

常用的高压缩率视频容器格式,包括*.mp4 、*.mkv、*.webM等。     容器格式本身并不直接决定压缩率,而是取决于容器中所使用的视频编码格式等因素。不过,在常见的视频容器格式中,一些容器在搭配特定编码格式时,通常能表现出较高的压缩效率,以下是相关介绍: 1 MKV格式 …

计算机视觉:主流数据集整理

第一章&#xff1a;计算机视觉中图像的基础认知 第二章&#xff1a;计算机视觉&#xff1a;卷积神经网络(CNN)基本概念(一) 第三章&#xff1a;计算机视觉&#xff1a;卷积神经网络(CNN)基本概念(二) 第四章&#xff1a;搭建一个经典的LeNet5神经网络(附代码) 第五章&#xff1…

SpringCloud系列教程:微服务的未来(二十五)-基于注解的声明队列交换机、消息转换器、业务改造

前言 在现代分布式系统中&#xff0c;消息队列是实现服务解耦和异步处理的关键组件。Spring框架提供了强大的支持&#xff0c;使得与消息队列&#xff08;如RabbitMQ、Kafka等&#xff09;的集成变得更加便捷和灵活。本文将深入探讨如何利用Spring的注解驱动方式来配置和管理队…

WordPress Elementor提示错误无法保存500的解决指南

500内部服务器错误是一种常见的服务器错误&#xff0c;通常由网站的服务器环境引起。这种错误可能导致网站无法正常访问&#xff0c;影响用户体验。本文将探讨500错误的常见原因&#xff0c;并提供解决方案&#xff0c;特别针对使用Elementor构建的WordPress网站。 500错误的常…

【WebGL】fbo双pass案例

双pass渲染案例&#xff08;离线渲染一个三角面&#xff0c;然后渲染到一个占满屏幕的矩阵上&#xff09; 离线渲染如何需要开启深度测试的话&#xff0c;需要额外操作&#xff0c;这里不展开 <!DOCTYPE html> <html lang"en"><head><meta ch…