Java的反射是指在程序运行时,对于任意一个类,都可以获取到这个类的所有属性和方法,并能够对其进行操作。通过反射机制,可以在程序运行时动态地创建对象、调用方法、获取属性值等。反射可以帮助我们更轻松地实现一些复杂的功能,但也会带来一些性能上的损失。
Java反射机制的基本步骤:
示例代码:
Class clazz = Class.forName("java.lang.String"); // 获取String类的Class对象
示例代码:
Object obj = clazz.newInstance(); // 实例化一个String对象
示例代码:
Field field = clazz.getField("value"); // 获取value属性
Object value = field.get(strObj); // 获取value属性的值
field.set(strObj, newValue); // 设置value属性的值为newValue
示例代码:
Method method = clazz.getMethod("substring", int.class, int.class); // 获取substring方法
Object result = method.invoke(strObj, startIndex, endIndex); // 调用substring方法,传递参数startIndex和endIndex
示例代码:
Constructor constructor = clazz.getConstructor(String.class); // 获取含有一个String类型参数的构造函数
Object obj = constructor.newInstance("Hello, World!"); // 调用构造函数并实例化对象
Java的反射机制带来了很多便利,但同时也会损失一些性能,在使用时需要慎重考虑其是否真正必要。
像咱们平时大部分时候都是在写业务代码,很少会接触到直接使用反射机制的场景。但是!这并不代表反射没有用。相反,正是因为反射,你才能这么轻松地使用各种框架。像 Spring/Spring Boot、MyBatis 等等框架中都大量使用了反射机制。
这些框架中也大量使用了动态代理,而动态代理的实现也依赖反射。
比如下面是通过 JDK 实现动态代理的示例代码,其中就使用了反射类 Method 来调用指定的方法。
public class DebugInvocationHandler implements InvocationHandler {
/**
* 代理类中的真实对象
*/
private final Object target;
public DebugInvocationHandler(Object target) {
this.target = target;
}
public Object invoke(Object proxy, Method method, Object[] args) throws InvocationTargetException, IllegalAccessException {
System.out.println("before method " + method.getName());
Object result = method.invoke(target, args);
System.out.println("after method " + method.getName());
return result;
}
}
另外,像 Java 中的一大利器 注解 的实现也用到了反射。为什么你使用 Spring 的时候 ,一个@Component注解就声明了一个类为 Spring Bean 呢?为什么你通过一个 @Value注解就读取到配置文件中的值呢?究竟是怎么起作用的呢?这些都是因为你可以基于反射分析类,然后获取到类/属性/方法/方法的参数上的注解。你获取到注解之后,就可以做进一步的处理。
反射在Java中是一种高级的语言特性,它允许程序在运行时动态地获取类的信息、访问和操作类的属性和方法等。因此,它的应用场景非常广泛,以下是一些例子:
框架开发:许多框架(如Spring)利用反射获取类的信息和属性,来实现依赖注入、控制反转等功能。
动态代理:利用反射机制来动态生成代码,从而实现对某些对象的代理。
注解解析:通过反射来获取类、方法、字段等的注解信息。
Java序列化:反射机制使得可以在编译时未知的情况下,将对象序列化为字节流,然后再通过反射将字节流转换成对象。
反编译工具:反编译工具利用反射实现对Java代码的分析和探测,可以查看方法、字段的信息以及代码的执行流程等。
总之,反射机制是Java语言中非常有用的一种特性,能够极大地提高程序的可扩展性、灵活性和可维护性。
总之,优点就是动态获取类信息,操作类的属性和方法们这是核心,缺点无非就是性能、安全和可读性,一定不要死记硬背,要通过原理去理解。
Class clz = Class.forName("java.lang.String");
Class clz = String.class;
String str = new String("Hello");
Class clz = str.getClass();
除了上述几种种方法,还可以通过ClassLoader类的loadClass()方法来获取Class对象,这种方式常用于加载外部或动态生成的类。
反射 API 用来生成 JVM 中的类、接口或则对象的信息。
Class 类:反射的核心类,可以获取类的属性,方法等信息。
Field 类:Java.lang.reflect 包中的类,表示类的成员变量,可以用来获取和设置类之中的属性值。
Method 类:Java.lang.reflect 包中的类,表示类的方法,它可以用来获取类中的方法信息或者执行方法。
Constructor 类:Java.lang.reflect 包中的类,表示类的构造方法。
获取想要操作的类的Class对象,这是反射的核心,通过Class对象我们可以任意调用类的方法。
调用 Class 类中的方法,既就是反射的使用阶段。
使用反射 API 来操作这些信息。
具体可以看下面的例子:
public class Student {
private Integer age;
public Integer getAge() {
return age;
}
public void setAge(Integer age) {
this.age = age;
}
public static void main(String[] args) {
//正常调用
Student student = new Student();
student.setAge(16);
System.out.println("Student Age : " + student.getAge());
//使用反射调用
try {
Class<?> aClass = Class.forName("com.lzl.javaSE.test8.Student");
Method setAge = aClass.getMethod("setAge", Integer.class);
Constructor<?> studentConstructor = aClass.getConstructor();
Object studentObject = studentConstructor.newInstance();
setAge.invoke(studentObject, 18);
Method getAge = aClass.getMethod("getAge");
System.out.println("Student Age : " + getAge.invoke(studentObject));
} catch (ClassNotFoundException | InstantiationException | InvocationTargetException | NoSuchMethodException |
IllegalAccessException e) {
throw new RuntimeException(e);
}
}
}
从代码中可以看到我们使用反射调用了 setAge方法,并传递了 16 的值。之后使用反射调用了 getAge 方法,输出其年龄。上面的代码整个的输出结果是:
Student Age : 16
Student Age : 18
从这个简单的例子可以看出,一般情况下我们使用反射获取一个对象的步骤:
Class<?> aClass = Class.forName("com.lzl.javaSE.test8.Student");
Constructor<?> studentConstructor = aClass.getConstructor();
Object studentObject = studentConstructor.newInstance();
而如果要调用某一个方法,则需要经过下面的步骤:
Method setAge = aClass.getMethod("setAge", Integer.class);
setAge.invoke(studentObject, 18);
Java通过反射机制可以获取和操作类、方法、字段等信息。在Java中,私有属性对外不可见,因此不能直接访问私有属性的值。但是,我们可以使用反射机制来绕过访问限制,获取私有属性的值。
下面是获取私有属性值的示例代码:
import java.lang.reflect.Field;
public class PrivateFieldTest {
private String privateField = "private value";
public static void main(String[] args) throws Exception {
PrivateFieldTest obj = new PrivateFieldTest();
// 获取私有属性对象
Field field = PrivateFieldTest.class.getDeclaredField("privateField");
// 设置可访问私有属性
field.setAccessible(true);
// 获取私有属性值
String value = (String) field.get(obj);
// 输出私有属性值
System.out.println("privateField value: " + value);
}
}
在上面的示例代码中,我们首先定义了一个私有属性privateField,然后在main方法中通过反射机制获取privateField的值。这里要注意,我们要通过类的getDeclaredField方法获取私有属性对象,并且要将其设置为可访问状态。
另外,我们还需要调用Field类的get方法获取私有属性的值,其中第一个参数是对象实例,表示要获取该对象中的私有属性值。最后我们将获取到的私有属性值输出。