浅谈Java内省

京东云开发者
• 阅读 18

作者:京东物流 王国泰

一、什么是内省

  • 讲内省,不得不说Java Bean,Bean在Java中是一种特殊的类,主要用于装载数据,数据会被存储在类的私有属性中,通常具有无参构造函数、可序列化、以及通过getter和setter方法来访问属性。内省是Java Beans规范的一部分,使用java.beans包中的类来实现,最常用的类是Introspector。通过内省,你可以获取一个Java Bean的属性描述符(PropertyDescriptor)和方法描述符(MethodDescriptor)

二、内省常用API

1、相关类

浅谈Java内省

2、Introspector

2.1 核心功能

  • 用于获取Bean的整体信息,包括属性描述符、方法描述符和事件描述符等

2.2 核心方法

  • getBeanInfo
BeanInfo beanInfo = Introspector.getBeanInfo(Vehicle.class);

3、BeanInfo

3.1 核心功能

  • 用于提供有关Bean的元数据,通常用于描述Bean的属性、事件和方法

3.2 核心方法

  • getPropertyDescriptors
PropertyDescriptor[] propertyDescriptors = beanInfo.getPropertyDescriptors();
  • getMethodDescriptors
MethodDescriptor[] methodDescriptors = beanInfo.getMethodDescriptors();
  • getEventSetDescriptors
EventSetDescriptor[] eventSetDescriptors = beanInfo.getEventSetDescriptors();

4、PropertyDescriptor

4.1 核心功能

  • 用于描述Bean的属性,提供了对属性的详细描述,包括属性的名称、类型、读方法getter、写方法setter等

4.2 核心方法

  • getName
PropertyDescriptor namePD = new PropertyDescriptor("name", Vehicle.class);
String name = namePD.getName();
  • getReadMethod
PropertyDescriptor namePD = new PropertyDescriptor("name", Vehicle.class);
Method getter = namePD.getReadMethod();

String methodName = getter.getName();
String vehicleName = (String) getter.invoke(new Vehicle());
  • getWriteMethod
PropertyDescriptor namePD = new PropertyDescriptor("name", Vehicle.class);
Method setter = namePD.getWriteMethod();

String methodName = setter.getName();
setter.invoke(new Vehicle(), "JD0001");

5、MethodDescriptor

5.1 核心功能

  • 用于描述一个方法的属性,提供了对方法的详细描述,包括方法的名称、参数类型、返回类型等

5.2 核心方法

  • getName
MethodDescriptor methodDescriptor = new MethodDescriptor(Vehicle.class.getMethod("setName", String.class));
String name = methodDescriptor.getName();
  • getMethod
MethodDescriptor methodDescriptor = new MethodDescriptor(Vehicle.class.getMethod("setName", String.class));
Method method = methodDescriptor.getMethod();
method.invoke(new Vehicle(), "JD0001");

6、EventSetDescriptor

6.1 核心功能

  • 用于描述一个Bean能够触发的一组事件,提供了有关事件监听器类型、添加和移除监听器的方法以及事件通知方法的信息

6.2 核心方法

  • 不常用

三、内省常见使用场景

1、依赖注入

  • Spring使用内省来分析类的构造函数、字段和方法,并自动注入依赖对象,可参考BeanWrapperImpl,部分源码如下:
@Override
public PropertyDescriptor[] getPropertyDescriptors() {
  return getCachedIntrospectionResults().getPropertyDescriptors();
}

2、对象拷贝

  • Spring BeanUtils使用内省来复制对象的属性,可参考BeanUtils,部分源码如下:
public static PropertyDescriptor[] getPropertyDescriptors(Class<?> clazz) throws BeansException {
  return CachedIntrospectionResults.forClass(clazz).getPropertyDescriptors();
}

3、开发工具和IDE

  • 开发工具和集成开发环境(IDE,如IntelliJ IDEA)使用内省来提供代码补全、重构、调试等功能

四、内省优缺点

1、优点

  • 灵活性和可扩展性:允许在运行时动态地获取和操作对象的属性和方法
  • 简化开发工作:支持框架和工具的开发,能够自动处理对象的属性和方法

2、缺点

  • 性能开销:比直接调用方法或访问字段要慢,而且不当使用可能会导致内存泄漏或增加GC压力
  • 访问安全:绕过Java的访问控制机制,访问私有字段和方法,可能会带来安全隐患,特别是在处理敏感数据时
  • 类型安全:通常是基于字符串名称进行的(如方法名、属性名),在编译时无法检查其正确性,容易导致运行时错误
  • 可读性和可维护性:代码可读性差,增加调试难度

五、内省与反射的区别

1、用途

  • 内省主要用于Java Bean的属性操作,适合于标准化的Bean操作
  • 反射则是更通用的机制,可以操作类的所有成员,包括私有成员

2、实现

  • 内省是基于Java Beans规范的,使用java.beans包
  • 反射是Java语言的核心特性,使用java.lang.reflect包

3、性能

  • 内省通常比反射快,主要原因是内省使用了缓存机制,减少了权限检查,并且在设计上针对特定场景进行了优化

相关文献

点赞
收藏
评论区
推荐文章
Wesley13 Wesley13
3年前
java面试 (六)
1 String.split(Stringregex),传入的参数是正则表达式,有一些特殊字符(比如.\\()\\|等)需要转义。2 关于枚举类型,一般用作常量,理想情况下,枚举中的属性字段是私有的,并在私有构造函数中赋值,没有对应的Setter方法,最好加上final修饰符。publicenumPer
Wesley13 Wesley13
3年前
1.利用BeanMap进行对象与Map的相互转换
javabean与map的转换有很多种方式,比如:1、通过ObjectMapper先将bean转换为json,再将json转换为map,但是这种方法比较绕,且效率很低,经测试,循环转换10000个bean,就需要12秒!!!不推荐使用2、通过java反射,获取bean类的属性和值,再转换到map对应的键值对中,这种方法次之,但稍微有点麻烦3、通过
Stella981 Stella981
3年前
Json解析工具 @JsonIgnore 注解的使用
@JsonIgnoreProperties        此注解是类注解,作用是json序列化时将javabean中的一些属性忽略掉,序列化和反序列化都受影响。@JsonIgnore        此注解用于属性或者方法上(最好是属性上),作用和上面的@JsonIgnoreProperties一样。@JsonFo
Wesley13 Wesley13
3年前
Java类和对象
一、类类是封装对象的属性和行为的载体,在Java语言中对象的属性以成员变量的形式存在,而对象的方法以成员方法的形式存在。1\.类的构造方法构造方法是一个与类同名的方法,对象的创建就是通过构造方法完成的,构造方法分为有参构造方法和无参构造方法,区别就在于有没有参数。说这么多概念是不是感觉有点麻木,直接看下面的例子吧。pub
Stella981 Stella981
3年前
Spring 4.3 的新功能和增强
核心容器改进核心容器额外提供了更丰富的元数据来改进编程。默认Java8的方法检测为bean属性的getter/setter方法。如果目标bean只定义了一个构造函数,则它无需要指定@Autowired注解@Configuration类支持构造函数注入。任何SpEL
Stella981 Stella981
3年前
Idea插件lombok的安装和使用
  C在写一个实体类时,有属性的写法,省去了写getter和setter的麻烦。  在Java编程时,写完字段后,需要一个一个去写getter和setter方法。在使用Idea编程时,可以按住ALTINSERT键,让IDE自动生成getter,setter,toString等方法,如下图:!(https://static.oschina.
Wesley13 Wesley13
3年前
Java基础之内省
Java基础之内省什么是内省  首先,我们要知道什么是内省。按我自己的理解就是在反射的原理上进行封装,来更方便的操作JavaBeanJavaBean就是特殊格式的类,其规范为:1.JavaBean类必须是一个公共类,即使用关键字public声明类。2.JavaBean类中必须有
Wesley13 Wesley13
3年前
Java反射之对JavaBean的内省操作
上一篇我们说了Java反射之数组的反射应用(https://www.oschina.net/action/GoToLink?urlhttps%3A%2F%2Fblog.csdn.net%2FOooops_%2Farticle%2Fdetails%2F100176965)这篇我们来模拟实现那些javabean的框架(BeanUtils)的基本操
小万哥 小万哥
10个月前
深入剖析 Java 类属性与类方法的应用
Java类属性Java类属性,也称为字段,是类中的变量。它们用于存储与类相关的数据。创建类属性在类定义中声明属性:javapublicclassMainintx;//属性Stringname;//属性访问类属性使用点语法访问对象的属性:javaMainmy
小万哥 小万哥
10个月前
Java 构造函数与修饰符详解:初始化对象与控制权限
Java构造函数Java构造函数是一种特殊的类方法,用于在创建对象时初始化对象的属性。它与类名相同,并且没有返回值类型。构造函数的作用:为对象的属性设置初始值执行必要的初始化操作提供创建对象的多种方式构造函数的类型:默认构造函数:无参数的构造函数,如果用户