何为反射?
在Java中,反射是指通过运行时的机制来操作类或对象的属性、方法和构造函数的能力。Java中的反射API提供了一组类和接口,使开发者可以在运行时获取和使用类的信息,而不需要提前知道它们的类型。
通过反射,我们可以在运行时获取和操作类的属性、方法和构造函数,即使在编译时无法提前知道它们的存在。这使得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 中的一大利器注解的实现也用到了反射。这些都是因为我们可以基于反射分析类,然后获取到类/属性/方法/方法的参数上的注解。获取到注解之后,就可以做进一步的处理。
谈谈反射机制的优缺点:
优点:可以让咱们的代码更加灵活、为各种框架提供开箱即用的功能提供了便利
缺点:让我们在运行时有了分析操作类的能力,这同样也增加了安全问题。比如可以无视泛型参数的安全检查(泛型参数的安全检查发生在编译时)。另外,反射的性能也要稍差点,不过,对于框架来说实际是影响不大的。