基本概念

Wiki 中是这样描述内省的:

  1. 在计算机科学中,内省是指计算机程序在运行时(Run time)检查对象(Object)类型的一种能力,通常也可以称作运行时类型检查。

这个描述非常宽泛,但有三个关键词:

  • 运行时
  • 对象
  • 类型

Java 官方对 Java Beans 内省的定义:

  1. At runtime and in the builder environment we need to be able to figure out which properties, events, and methods a Java Bean supports. We call this process introspection.

从 Java Bean 的角度来看,这里的对象就是 Bean 对象,主要关注点是属性、方法和事件等,也就是说在运行时可以获取相应的信息进行一些处理,这就是 Java Beans 的内省机制。

与反射的区别

Java Beans 内省其实就是对反射的一种封装,这个从源码中或者官方文档中都能看到:

  1. By default we will use a low level reflection mechanism to study the methods supported by a target bean and then apply simple design patterns to deduce from those methods what properties, events, and public methods are supported.

Java Beans 内省机制

核心类库

Java Beans 内省机制的核心类是 Introspector:

  1. * The Introspector class provides a standard way for tools to learn about
  2. * the properties, events, and methods supported by a target Java Bean.

操作范围主要包括但不局限于 Java Beans 的属性,事件和方法,具体是基于以下几个类实现:

  • BeanInfo
    • Java Bean 信息类
  • PropertyDescriptor
    • 属性描述类
  • MethodDescriptor
    • 方法描述类
  • EventSetDescriptor
    • 事件描述集合

先看一个示例:
定义一个 Java Bean:

  1. public class User {
  2. private String username;
  3. private Integer age;
  4. // getter/setter
  5. // toString
  6. }

测试代码:

  1. @Test
  2. public void test1() throws IntrospectionException {
  3. //获取 User Bean 信息
  4. BeanInfo userBeanInfo = Introspector.getBeanInfo(User.class);
  5. //属性描述
  6. PropertyDescriptor[] propertyDescriptors = userBeanInfo.getPropertyDescriptors();
  7. System.out.println("属性描述:");
  8. Stream.of(propertyDescriptors).forEach(System.out::println);
  9. //方法描述
  10. System.out.println("方法描述:");
  11. MethodDescriptor[] methodDescriptors = userBeanInfo.getMethodDescriptors();
  12. Stream.of(methodDescriptors).forEach(System.out::println);
  13. //事件描述
  14. System.out.println("事件描述:");
  15. EventSetDescriptor[] eventSetDescriptors = userBeanInfo.getEventSetDescriptors();
  16. Stream.of(eventSetDescriptors).forEach(System.out::println);
  17. }

输出结果:

  1. 属性描述:
  2. java.beans.PropertyDescriptor[name=age; propertyType=class java.lang.Integer; readMethod=public java.lang.Integer introspector.bean.User.getAge(); writeMethod=public void introspector.bean.User.setAge(java.lang.Integer)]
  3. java.beans.PropertyDescriptor[name=class; propertyType=class java.lang.Class; readMethod=public final native java.lang.Class java.lang.Object.getClass()]
  4. java.beans.PropertyDescriptor[name=username; propertyType=class java.lang.String; readMethod=public java.lang.String introspector.bean.User.getUsername(); writeMethod=public void introspector.bean.User.setUsername(java.lang.String)]
  5. 方法描述:
  6. java.beans.MethodDescriptor[name=getClass; method=public final native java.lang.Class java.lang.Object.getClass()]
  7. java.beans.MethodDescriptor[name=setAge; method=public void introspector.bean.User.setAge(java.lang.Integer)]
  8. java.beans.MethodDescriptor[name=getAge; method=public java.lang.Integer introspector.bean.User.getAge()]
  9. java.beans.MethodDescriptor[name=wait; method=public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException]
  10. java.beans.MethodDescriptor[name=notifyAll; method=public final native void java.lang.Object.notifyAll()]
  11. java.beans.MethodDescriptor[name=notify; method=public final native void java.lang.Object.notify()]
  12. java.beans.MethodDescriptor[name=getUsername; method=public java.lang.String introspector.bean.User.getUsername()]
  13. java.beans.MethodDescriptor[name=wait; method=public final void java.lang.Object.wait() throws java.lang.InterruptedException]
  14. java.beans.MethodDescriptor[name=hashCode; method=public native int java.lang.Object.hashCode()]
  15. java.beans.MethodDescriptor[name=setUsername; method=public void introspector.bean.User.setUsername(java.lang.String)]
  16. java.beans.MethodDescriptor[name=wait; method=public final native void java.lang.Object.wait(long) throws java.lang.InterruptedException]
  17. java.beans.MethodDescriptor[name=equals; method=public boolean java.lang.Object.equals(java.lang.Object)]
  18. java.beans.MethodDescriptor[name=toString; method=public java.lang.String introspector.bean.User.toString()]
  19. 事件描述:

可以看出通过内省机制可以获取 Java Bean 的属性、方法描述,这里事件描述是空的(关于事件相关会在后面介绍)。由于 Java 类都会继承 Object 类,可以看到这里将 Object 类相关的属性和方法描述也输出了,如果想将某个类的描述信息排除可以使用 java.beans.Introspector#getBeanInfo(java.lang.Class<?>, java.lang.Class<?>) 这个方法。

属性处理

配置绑定

通过 PropertyDescriptor 可以基于字段名为可写属性设置值。
比如我们经常会使用这样的配置文件:

  1. user:
  2. username: zhangsan
  3. age: 1

配置文件会与对象进行数据绑定。测试代码:

  1. @Test
  2. public void test2() throws IntrospectionException {
  3. YamlPropertiesFactoryBean yaml = new YamlPropertiesFactoryBean();
  4. yaml.setResources(new ClassPathResource("application.yml"));
  5. String path = "user.";
  6. Properties properties = yaml.getObject();
  7. System.out.println(properties);
  8. User user = new User();
  9. //获取 User Bean 信息,排除 Object
  10. BeanInfo userBeanInfo = Introspector.getBeanInfo(User.class, Object.class);
  11. //属性描述
  12. PropertyDescriptor[] propertyDescriptors = userBeanInfo.getPropertyDescriptors();
  13. Stream.of(propertyDescriptors).forEach(propertyDescriptor -> {
  14. //获取属性名称
  15. String property = propertyDescriptor.getName();
  16. try {
  17. propertyDescriptor.getWriteMethod().invoke(user,properties.get(path+property));
  18. } catch (IllegalAccessException | InvocationTargetException ignored) {
  19. }
  20. });
  21. System.out.println(user);
  22. }

输出结果:

  1. User{username='zhangsan', age=1}

在 Spring 中的使用

在传统的 Spring 开发中我们需要在 web.xml 中指定一些配置参数,比如:

  1. <servlet>
  2. <servlet-name>app</servlet-name>
  3. <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
  4. <init-param>
  5. <param-name>contextConfigLocation</param-name>
  6. <param-value></param-value>
  7. </init-param>
  8. <load-on-startup>1</load-on-startup>
  9. </servlet>

这里有一个 contextConfigLocation 参数,这个参数最终是与 FrameworkServlet 类中的一个属性进行绑定:

  1. public abstract class FrameworkServlet extends HttpServletBean implements ApplicationContextAware {
  2. private String contextConfigLocation;
  3. }

那么 Spring 是如何将 web.xml 中的配置项与属性进行绑定的呢,可以参数看org.springframework.web.servlet.HttpServletBean#init() 方法:

  1. @Override
  2. public final void init() throws ServletException {
  3. // Set bean properties from init parameters.
  4. PropertyValues pvs = new ServletConfigPropertyValues(getServletConfig(), this.requiredProperties);
  5. if (!pvs.isEmpty()) {
  6. try {
  7. BeanWrapper bw = PropertyAccessorFactory.forBeanPropertyAccess(this);
  8. ResourceLoader resourceLoader = new ServletContextResourceLoader(getServletContext());
  9. bw.registerCustomEditor(Resource.class, new ResourceEditor(resourceLoader, getEnvironment()));
  10. initBeanWrapper(bw);
  11. bw.setPropertyValues(pvs, true);
  12. }
  13. catch (BeansException ex) {
  14. if (logger.isErrorEnabled()) {
  15. logger.error("Failed to set bean properties on servlet '" + getServletName() + "'", ex);
  16. }
  17. throw ex;
  18. }
  19. }
  20. // Let subclasses do whatever initialization they like.
  21. initServletBean();
  22. }

可以看到 Spring 是通过 BeanWrapper 完成对属性的绑定:

  1. public interface BeanWrapper extends ConfigurablePropertyAccessor {
  2. // 获取属性描述器
  3. PropertyDescriptor[] getPropertyDescriptors();
  4. PropertyDescriptor getPropertyDescriptor(String var1) throws InvalidPropertyException;
  5. }

而 BeanWrapper 又继承了 PropertyAccessor 接口:

  1. public interface PropertyAccessor {
  2. //读属性
  3. boolean isReadableProperty(String var1);
  4. //写属性
  5. boolean isWritableProperty(String var1);
  6. @Nullable
  7. Class<?> getPropertyType(String var1) throws BeansException;
  8. @Nullable
  9. TypeDescriptor getPropertyTypeDescriptor(String var1) throws BeansException;
  10. }

也就是说 Spring 中 BeanWrapper 基于 Java 的内省机制实现了对属性的赋值工作,但是 Spring 并未局限于 Java 提供的 API,而是也进行了扩展和进一步的封装,如 TypeDescriptor。
可以参考 org.springframework.web.servlet.HttpServletBean#init() 中 BeanWrapper 的使用来实现对 User 对象的属性赋值:

  1. @Test
  2. public void test5(){
  3. User user = new User();
  4. BeanWrapper bw = PropertyAccessorFactory.forBeanPropertyAccess(user);
  5. MutablePropertyValues pvs = new MutablePropertyValues();
  6. pvs.add("username","zhangsan");
  7. pvs.add("age",1);
  8. bw.setPropertyValues(pvs);
  9. System.out.println(user);
  10. }

输出结果:

  1. User{username='zhangsan', age=1}

类型转换

有属性赋值,必然就会有类型转换。说白了我们从配置文件读取的数据是字符串,与属性进行参数绑定的过程中势必会有类型转换,java.beans 中提供了相应的 API:

  • PropertyEditor
    • 属性编辑器顶层接口
  • PropertyEditorSupport
    • 属性编辑器实现类
  • PropertyEditorManager
    • 属性编辑器管理器
    • 在 Spring 中提供了一个 PropertyEditorRegistrar

先看一个例子:
User 类增加 Date 属性:

  1. public class User {
  2. private String username;
  3. private Integer age;
  4. private Date createTime;
  5. // getter/setter
  6. // toString
  7. }

日期转换器:

  1. /**
  2. * 日期属性编辑器
  3. */
  4. public class DatPropertyEditor extends PropertyEditorSupport {
  5. @Override
  6. public void setAsText(String text) {
  7. try {
  8. setValue((text == null) ? null : new SimpleDateFormat("yyyy-MM-dd").parse(text));
  9. } catch (ParseException e) {
  10. e.printStackTrace();
  11. }
  12. }
  13. }

在之前的例子中内省设置属性值都是直接通过 PropertyDescriptor 获取属性的写方法通过反射去赋值,而如果需要对值进行类型转换,则需要通过 PropertyEditorSupport#setAsText 调用 setValue 方法,然后 setValue 方法触发属性属性修改事件:

  1. public class PropertyEditorSupport implements PropertyEditor {
  2. public void setValue(Object value) {
  3. this.value = value;
  4. firePropertyChange();
  5. }
  6. }

要注意这里的 value 实际上是临时存储在 PropertyEditorSupport 中,PropertyEditorSupport 则作为事件源,从而得到类型转换后的 value,再通过 PropertyDescriptor 获取属性的写方法通过反射去赋值。
测试代码:

  1. @Test
  2. public void test6() throws IntrospectionException, FileNotFoundException {
  3. Map<String,Object> properties = ImmutableMap.of("age",1,"username","zhangsan","createTime","2020-01-01");
  4. User user = new User();
  5. //获取 User Bean 信息,排除 Object
  6. BeanInfo userBeanInfo = Introspector.getBeanInfo(User.class, Object.class);
  7. //属性描述
  8. PropertyDescriptor[] propertyDescriptors = userBeanInfo.getPropertyDescriptors();
  9. Stream.of(propertyDescriptors).forEach(propertyDescriptor -> {
  10. //获取属性名称
  11. String property = propertyDescriptor.getName();
  12. //值
  13. Object value = properties.get(property);
  14. if (Objects.equals("createTime", property)) {
  15. //设置属性编辑器
  16. propertyDescriptor.setPropertyEditorClass(DatPropertyEditor.class);
  17. //创建属性编辑器
  18. PropertyEditor propertyEditor = propertyDescriptor.createPropertyEditor(user);
  19. //添加监听器
  20. propertyEditor.addPropertyChangeListener(evt -> {
  21. //获取转换后的value
  22. Object value1 = propertyEditor.getValue();
  23. setPropertyValue(user, propertyDescriptor, value1);
  24. });
  25. propertyEditor.setAsText(String.valueOf(value));
  26. return;
  27. }
  28. setPropertyValue(user, propertyDescriptor, value);
  29. });
  30. System.out.println(user);
  31. }
  32. /**
  33. * 设置属性值
  34. */
  35. private void setPropertyValue(User user, PropertyDescriptor propertyDescriptor, Object value1) {
  36. try {
  37. propertyDescriptor.getWriteMethod().invoke(user, value1);
  38. } catch (IllegalAccessException | InvocationTargetException ignored) {
  39. }
  40. }

输出结果:

  1. User{username='zhangsan', age=1, createTime=2020-1-1 0:00:00}

事件监听

先举一个的例子,在 Mac 中设置飞书通知方式的时候,当界面右侧选择“提示”的时候,那么左侧也会相应的显示为“提示”:
image.png
如果将右侧看成一个 Java Bean,那么这中间势必存在一个属性变化监听。java.beans 包中也提供了相应实现:

  • PropertyChangeEvent
    • 属性变化事件
  • PropertyChangeListener
    • 属性(生效)变化监听器
  • PropertyChangeSupport
    • 属性(生效)变化监听器管理器’
  • VetoableChangeListener
    • 属性(否决)变化监听器
  • VetoableChangeSupport
    • 属性(否决)变化监听器管理器

PropertyChangeEvent 的构造方法:

  1. public PropertyChangeEvent(Object source, String propertyName,
  2. Object oldValue, Object newValue) {
  3. super(source);
  4. this.propertyName = propertyName;
  5. this.newValue = newValue;
  6. this.oldValue = oldValue;
  7. }

通过这个构造方法可以看出属性变化监听的关注点:

  • source
    • 事件源
  • propertyName
    • 发生变化的属性名称
  • oldValue
    • 旧值
  • newValue
    • 新值

示例代码:
在 User 中增加属性(生效)变化监听:

  1. public class User {
  2. private String username;
  3. private Integer age;
  4. /**
  5. * 属性(生效)变化监听器管理器
  6. */
  7. private PropertyChangeSupport propertyChangeSupport = new PropertyChangeSupport(this);
  8. /**
  9. * 启动属性(生效)变化
  10. * @param propertyName
  11. * @param oldValue
  12. * @param newValue
  13. */
  14. private void firePropertyChange(String propertyName, String oldValue, String newValue) {
  15. PropertyChangeEvent event = new PropertyChangeEvent(this, propertyName, oldValue, newValue);
  16. propertyChangeSupport.firePropertyChange(event);
  17. }
  18. /**
  19. * 添加属性(生效)变化监听器
  20. */
  21. public void addPropertyChangeListener(PropertyChangeListener listener){
  22. propertyChangeSupport.addPropertyChangeListener(listener);
  23. }
  24. /**
  25. * 删除属性(生效)变化监听器
  26. */
  27. public void removePropertyChangeListener(PropertyChangeListener listener){
  28. propertyChangeSupport.removePropertyChangeListener(listener);
  29. }
  30. /**
  31. * 获取属性(生效)变化监听器
  32. */
  33. public PropertyChangeListener[] getPropertyChangeListeners() {
  34. return propertyChangeSupport.getPropertyChangeListeners();
  35. }
  36. public void setUsername(String username) {
  37. String oldValue = this.username;
  38. this.username = username;
  39. firePropertyChange("username", oldValue, username);
  40. }
  41. // getter/setter
  42. // toString
  43. }

测试代码:

  1. @Test
  2. public void test3(){
  3. User user = new User();
  4. user.setAge(1);
  5. user.setUsername("zhangsan");
  6. user.addPropertyChangeListener(System.out::println);
  7. user.setUsername("lisi");
  8. user.setUsername("wangwu");
  9. }

输出结果:

  1. java.beans.PropertyChangeEvent[propertyName=name; oldValue=zhangsan; newValue=lisi; propagationId=null; source=User{username='lisi', age=1}]
  2. java.beans.PropertyChangeEvent[propertyName=name; oldValue=lisi; newValue=wangwu; propagationId=null; source=User{username='wangwu', age=1}]

可以看到在添加了监听器后,当 username 属性发生变化的时候会出发监听事件。
再看看另外一种监听器 VetoableChangeListener。在 User 中添加监听器:

  1. /**
  2. * 属性(否决)变化监听器
  3. */
  4. private VetoableChangeSupport vetoableChangeSupport = new VetoableChangeSupport(this);
  5. /**
  6. * 启动属性(否决)变化
  7. * @param propertyName
  8. * @param oldValue
  9. * @param newValue
  10. */
  11. private void fireVetoableChange(String propertyName, String oldValue, String newValue) throws PropertyVetoException {
  12. PropertyChangeEvent event = new PropertyChangeEvent(this, propertyName, oldValue, newValue);
  13. vetoableChangeSupport.fireVetoableChange(event);
  14. }
  15. /**
  16. * 添加属性(否决)变化监听器
  17. */
  18. public void addVetoableChangeListener(VetoableChangeListener listener){
  19. vetoableChangeSupport.addVetoableChangeListener(listener);
  20. }
  21. /**
  22. * 删除属性(否决)变化监听器
  23. */
  24. public void removeVetoableChangeListener(VetoableChangeListener listener){
  25. vetoableChangeSupport.removeVetoableChangeListener(listener);
  26. }
  27. public void setUsername(String username) throws PropertyVetoException {
  28. String oldValue = this.username;
  29. fireVetoableChange("username",oldValue,username);
  30. this.username = username;
  31. firePropertyChange("username", oldValue, username);
  32. }

测试代码:

  1. @Test
  2. public void test3() throws PropertyVetoException {
  3. User user = new User();
  4. user.setAge(1);
  5. user.addVetoableChangeListener(evt -> {
  6. System.out.println(evt.getNewValue()+",,"+evt.getOldValue());
  7. if (Objects.equals(evt.getNewValue(), evt.getOldValue())) {
  8. throw new PropertyVetoException("当前属性值未发生任何变化", evt);
  9. }
  10. });
  11. user.addPropertyChangeListener(System.out::println);
  12. user.setUsername("lisi");
  13. user.setUsername("zhangsan");
  14. user.setUsername("zhangsan");
  15. }

运行时发现一直无法抛出异常。查看源码发现 PropertyChangeSupport 和 VetoableChangeSupport 当新旧值相等时不会触发监听,于是修改测试代码:

  1. @Test
  2. public void test3() throws PropertyVetoException {
  3. User user = new User();
  4. user.setAge(1);
  5. user.addVetoableChangeListener(evt -> {
  6. System.out.println(evt.getNewValue()+",,"+evt.getOldValue());
  7. if (Objects.isNull(evt.getNewValue())) {
  8. throw new PropertyVetoException("username 不能为null", evt);
  9. }
  10. });
  11. user.addPropertyChangeListener(System.out::println);
  12. user.setUsername("lisi");
  13. user.setUsername(null);
  14. }

运行结果:

  1. lisi,,null
  2. java.beans.PropertyChangeEvent[propertyName=username; oldValue=null; newValue=lisi; propagationId=null; source=User{username='lisi', age=1}]
  3. null,,lisi
  4. java.beans.PropertyVetoException: username 不能为null
  5. at introspector.test.IntrospectorTest.lambda$test3$1(IntrospectorTest.java:78)
  6. at java.beans.VetoableChangeSupport.fireVetoableChange(VetoableChangeSupport.java:375)

可以发现当符合“否决”属性变化的条件时,会抛出 PropertyVetoException 异常阻断属性的变化。
在之前的示例中 userBeanInfo 输出的 EventSetDescriptor 为空,这是因为并未到 User 类中增加事件。现在再测试一下获取 EventSetDescriptor:

  1. @Test
  2. public void test1() throws IntrospectionException {
  3. BeanInfo userBeanInfo = Introspector.getBeanInfo(User.class, Object.class);
  4. EventSetDescriptor[] eventSetDescriptors = userBeanInfo.getEventSetDescriptors();
  5. Stream.of(eventSetDescriptors).forEach(System.out::println);
  6. }

输出结果:

  1. java.beans.EventSetDescriptor[name=propertyChange; inDefaultEventSet; listenerType=interface java.beans.PropertyChangeListener; getListenerMethod=public java.beans.PropertyChangeListener[] introspector.bean.User.getPropertyChangeListeners(); addListenerMethod=public void introspector.bean.User.addPropertyChangeListener(java.beans.PropertyChangeListener); removeListenerMethod=public void introspector.bean.User.removePropertyChangeListener(java.beans.PropertyChangeListener)]
  2. java.beans.EventSetDescriptor[name=vetoableChange; inDefaultEventSet; listenerType=interface java.beans.VetoableChangeListener; addListenerMethod=public void introspector.bean.User.addVetoableChangeListener(java.beans.VetoableChangeListener); removeListenerMethod=public void introspector.bean.User.removeVetoableChangeListener(java.beans.VetoableChangeListener)]