zoukankan      html  css  js  c++  java
  • Java你只知道反射,可能会不知道内省

    1. 前言

    在上一文中我们封装了一个Mybatis通用Mapper。为了获得实体类属性我使用了反射。大多数同学也第一感觉会用反射实现,其实还有一种技术也能实现,这就是内省(Introspector)。

    2. 什么是内省

    在计算机科学中,内省是指计算机程序在运行时(Runtime)检查对象(Object)类型的一种能力,通常也可以称作运行时类型检查。不应该将内省和反射混淆。相对于内省,反射更进一步,是指计算机程序在运行时(Runtime)可以访问、检测和修改它本身状态或行为的一种能力。

    Java中的内省是对JavaBean属性、的一种缺省处理方法。看了概念是不是有点懵逼,我也一样。所以我们写个例子来看看就知道了。写之前还要搞清楚JavaBean的定义;

    • 属性是私有的。

    • 有无参的public构造方法。

    • 对于这些属性有公开的getter/setter方法。

    但是如果有的类比较“调皮” ,有getter/setter没有对应的实体,也容易被内省到,但是会抛出异常。所以遵守规约是一个有利于我们使用内省。

    其实 add/remove、is 也认为是操作JavaBean属性的方法,所以我们要小心。

    3. Java内省操作

    JavaBean一般用来传递数据使用,我们数据库实体类就是一种典型的JavaBean。

    public class UserInfo implements Serializable {
    
        private static final long serialVersionUID = -8938650956516110149L;
        private Long userId;
        private String name;
        private Integer age;
        private String time;
    
        public Long getUserId() {
            return userId;
        }
    
        public void setUserId(Long userId) {
            this.userId = userId;
        }
    
        public String getName() {
            return name;
        }
    
        public void setName(String name) {
            this.name = name;
        }
    
        public Integer getAge() {
            return age;
        }
    
        public void setAge(Integer age) {
            this.age = age;
        }
    }
    

    请注意我特意没有给time属性设置getter/setter。接下来我就开始演示使用内省来操作实体了。

    Java中通过java.beans.Introspector来进行内省操作。常用的内省操作主要有下面这些,当然还有其它的附加类型。

    • 3.1 BeanInfo

    BeanInfo就是内省对JavaBean的一个整体描述。这里我只想获取UserInfo的描述信息,所以应该这么写:

    BeanInfo beanInfo = Introspector.getBeanInfo(UserInfo.class,Object.class);
    

    你可以通过 stop 、flag 两个属性来控制内省分析的深度。

    • 3.2 BeanDescriptor

    然后我们看看BeanDescriptor主要有什么:

    BeanDescriptor beanDescriptor = beanInfo.getBeanDescriptor();
    // cn.felord.kono.entity.UserInfo.class
    Class<?> beanClass = beanDescriptor.getBeanClass();
    // UserInfo
    String name = beanDescriptor.getName();
    

    原来是JavaBean的Class类型和名称。

    • 3.3 PropertyDescriptors

    PropertyDescriptor是描述JavaBean的成员属性的,我们打印来看看

    Stream.of(beanInfo.getPropertyDescriptors())
            .forEach(System.out::println);
    java.beans.PropertyDescriptor[name=age; propertyType=class java.lang.Integer; readMethod=public java.lang.Integer cn.felord.kono.entity.UserInfo.getAge(); writeMethod=public void cn.felord.kono.entity.UserInfo.setAge(java.lang.Integer)]
    java.beans.PropertyDescriptor[name=name; propertyType=class java.lang.String; readMethod=public java.lang.String cn.felord.kono.entity.UserInfo.getName(); writeMethod=public void cn.felord.kono.entity.UserInfo.setName(java.lang.String)]
    java.beans.PropertyDescriptor[name=userId; propertyType=class java.lang.Long; readMethod=public java.lang.Long cn.felord.kono.entity.UserInfo.getUserId(); writeMethod=public void cn.felord.kono.entity.UserInfo.setUserId(java.lang.Long)]

    原来PropertyDescriptor包含了成员属性的名称、类型、读的方法、写的方法。注意到没有居然不包含time属性,因为它没有getter/setter被忽略了。

    • 3.4 MethodDescriptors

    顾名思义,一定是描述JavaBean的方法的。这里放出一条打印结果。

    java.beans.MethodDescriptor[name=foo; method=public static void cn.felord.kono.entity.UserInfo.foo(java.lang.String)]

    包含了方法名称和方法对象(Method)。注意这里混进来了奇怪的方法 foo,是的!我随便写了一个方法。MethodDescriptors可以包含JavaBean下所有的方法,包括静态方法。当然受到内省深度的制约。

    • 3.5 EventSetDescriptors

    目前打印为空,JavaBean 事件发布订阅相关的一些范式,目前我还不知道什么作用。

    4. 总结

    Java反射是在运行时获取一个类的所有信息,可以操纵类的字段、方法、构造器等,功能非常强大。而内省其实就是反射的一个子集,基于反射实现。专门操作JavaBean的,只关注于JavaBean的属性、方法、事件的一些属性。了解了这个我觉得有必要把通用Mapper重构一下了。

  • 相关阅读:
    TreeMap Red-Black tree
    Java实现生产者消费者问题与读者写者问题
    一个对象占用多大内存
    MySQL索引背后的数据结构及算法原理
    Java并发编程与技术内幕:线程池深入理解
    Java Spring 中你不知道的注入方式
    面试中的排序算法总结
    JAVA反射
    StringBuilder与StringBuffer的区别
    Java多线程总结【转】
  • 原文地址:https://www.cnblogs.com/47Gamer/p/13682286.html
Copyright © 2011-2022 走看看