担心被问到-反射,我来给你壮壮胆

开发 前端
在Java中,想要运行时操作对象的方法、属性,其中有效的手段,就有反射。这也是框架最常使用的手段,因为框架在编译时并不知晓系统中有哪些类会被使用。

\[[386511]]

本文转载自微信公众号「狼王编程」,作者狼王。转载本文请联系狼王编程公众号。

这篇让我们聊聊什么是反射,为啥好多程序员听到反射就莫名的害怕呢,本文给你壮壮胆,其实没多少东西的!

在Java中,想要运行时操作对象的方法、属性,其中有效的手段,就有反射。这也是框架最常使用的手段,因为框架在编译时并不知晓系统中有哪些类会被使用。

Java中对象的类型信息在运行时由Class对象表示,Class对象是伴随类加载而实例化产生的,而反射的实现就围绕着Class对象。

有很多种方式可以获取类的Class对象,包括:

  1. Class<Object> c1 = Object.class; 
  2. Class<?> c2 = Class.forName("java.lang.Object"); 
  3. Class<?> c3 = new Object().getClass(); 

反射可以做什么?

1. 实例化对象

对象的实例化,可以通过new关键字直接实例化,也可以通过反射,例如:

  1. class.newInstance()  
  2. class.getConstructor(Class<?>... parameterTypes).newInstance(Object ... initargs) 

为什么需要用反射来进行对象实例化?

有不能使用new关键字直接实例化对象的场景,例如:Spring容器管理的Bean,只能通过类的全限定名加载类,然后反射实例化。

在不想使用new关键字的场景,目的是为了简化编程,使代码美观,可能大家经常看到类似的用法,例如:

  1. public static <T> T parseObject(String text, Class<T> clazz) { 
  2.   return parseObject(text, clazz, new Feature[0]); 
  1. /** 
  2. * 简单的复制出新类型对象 
  3. */ 
  4. public static <S, D> D map(S source, Class<D> destinationClass){ 
  5.   return mapper.map(source, destinationClass); 

2. 筛选合适的类

实际开发中,经常会有这种需要,如果类有某某特征,就进行某某操作。在Spring进行扫描的时候,我们会通过过滤器,来精细化控制bean的生成,包括:

  1. 根据isInstance(Object obj)判断是否实现某某接口或者继承特殊的父类;
  2. 根据isAnnotationPresent(Class annotationClass)判断是否被注解标记。

3. 方法调用

有些场景不能或者不合适直接调用方法,例如:我们处理HTTP请求,需要从URI映射到方法调用,如果我们能够穷举所有的URL到对象方法的映射关系,那么也没问题,但是无数的if条件判断,显然不是明智的选择。

通常我们先会获取资源对象,然后反射调用对象的方法。

  1. Method.invoke(Object obj, Object... args) 

如何获取对象的方法对象,也就是Method对象呢?Class类提供了以下实现:

  1. Method[] getMethods(); 
  2. Method[] getDeclaredMethods(); 
  3. Method getMethod(String name, Class<?>... parameterTypes); 
  4. Method getDeclaredMethod(String name, Class<?>... parameterTypes); 

这些方法可以分为两类:

  1. 方法签名中带有Declared的,会在当前类的所有方法中查找,但不会遍历父类。
  2. 不带Declared的会遍历所有父类,但只会查找public方法。

推荐使用工具类:org.apache.commons.lang3.reflect.MethodUtils 此类中包含遍历所有父类查找方法、当前类查找public方法或者反射执行方法的便捷操作。

4. 属性操作

  1. Class.getFields(), Class.getField(String),  
  2. Class.getDeclaredFields(), Class.getDeclaredField(String) 

命名规则同方法,推荐使用工具类:org.apache.commons.lang3.FieldUtils,进行读取或者赋值操作。

反射进行赋值有几点需要注意:

  1. 如果不为public类型,那么设置字段前必须通过Field.setAccessible(true)方法进行访问权限设置,不然会抛出异常:IllegalAccessException 。
  2. 如果字段为static类型,那么通过set方法进行赋值时,会忽略obj对象,因为静态字段属于类。
  3. 如果字段为final类型,不管是public还是 private,那么用set方法赋值时,只有setAccessible进行访问权限设置后,才能正确调用,不然会报异常:IllegalAccessException。但是对final字段进行set方法赋值时,尽管方法正常调用,但是并不会改变fianl字段的值。
  4. 如果字段为final static 类型,那么进行set方法赋值时,总会抛出IllegalAccessException异常。

最后如果对当前对象的所有方法,或者所有字段进行某种操作,那么推荐工具类:org.springframework.util.ReflectionUtils

4、总结

 

 

 

所以反射其实没那么可怕,消灭恐惧的最好办法就是面对恐惧,加油学起来吧!

 

责任编辑:武晓燕 来源: 狼王编程
相关推荐

2020-08-18 07:58:41

反射Spring框架

2017-10-25 14:08:10

数据库MySQL原理及优化

2023-03-21 08:10:18

2020-10-14 15:53:45

秒杀秒杀系统流量

2023-10-26 16:02:04

线程

2015-08-04 09:24:50

2017-09-19 08:29:51

SSD寿命驱动器

2017-12-18 17:21:56

AndroidJava内存泄漏

2024-03-06 08:00:56

javaAQS原生

2012-08-03 09:14:23

2015-04-30 14:05:18

Visual Stud

2015-11-20 18:51:29

2022-07-27 11:05:31

面试经验

2023-04-10 09:32:00

DubboJava

2024-04-02 09:58:58

Java并发锁开发

2020-07-22 10:30:35

机器人人工智能系统

2016-04-28 09:36:44

人才教育/华三

2020-09-15 10:25:13

Redis命令Java

2021-04-30 09:04:11

Go 语言结构体type
点赞
收藏

51CTO技术栈公众号