内省(Introspector)
欢迎光临我的博客[http://poetize.cn],前端使用Vue2,聊天室使用Vue3,后台使用Spring Boot
内省(Introspector) 是Java语言对 JavaBean 类属性、事件的一种缺省处理方法。用来访问某个属性的 getter/setter 方法。
内省大部分是在写一些框架或者工具的时候会用到。比如说 spring 初始化 bean。
JavaBean
JavaBean是一种特殊的类,主要用于传递数据信息。
这种类中的方法主要用于访问私有的字段,且方法名符合某种命名规则。
如果在两个模块之间传递信息,可以将信息封装进JavaBean中,这种对象称为"值对象"(Value Object),或"VO"。
这些信息储存在类的私有变量中,通过set()、get()获得。
为写成JavaBean,类必须是具体的和公共的,并且具有无参数的构造器。
Java JDK中提供了一套 API 用来访问某个属性的 getter/setter 方法,这就是内省(Introspector)。
Introspector类
调用Introspector.getBeanInfo()方法,得到的BeanInfo对象(封装了这个类的属性的信息)。
User user=new User();
String userName="userName";
BeanInfo beanInfo=Introspector.getBeanInfo(User.class);
PropertyDescriptor[] proDescrtptors=beanInfo.getPropertyDescriptors();
if(proDescrtptors!=null&&proDescrtptors.length>0){
for(PropertyDescriptor propDesc:proDescrtptors){
if(propDesc.getName().equals(userName)){
Method methodSetUserName=propDesc.getWriteMethod();
methodSetUserName.invoke(user, "alan");
break;
}
}
JDK内省类库:PropertyDescriptor类
直接构造 PropertyDescriptors
User user = new User("jack", 21);
String propertyName = "name";
PropertyDescriptor namePd = new PropertyDescriptor(propertyName, User.class);
namePd.getWriteMethod().invoke(user, "tom");
namePd.getReadMethod().invoke(user);
主要方法:
1. getPropertyType(),获得属性的Class对象;
2. getReadMethod(),获得用于读取属性值的方法;getWriteMethod(),获得用于写入属性值的方法;
3. hashCode(),获取对象的哈希值;
4. setReadMethod(Method readMethod),设置用于读取属性值的方法;
5. setWriteMethod(Method writeMethod),设置用于写入属性值的方法。
BeanUtils基本操作
Introspector操作较繁琐,所以所以Apache开发了一套简单、易用的API来操作Bean的属性——BeanUtils工具包:
BeanUtils工具本身也是一种内省的实现方法,所以也是借助于底层的getter和setter方法进行转换的。
BeanUtils设置属性值的时候,如果属性是基本数据类型,那么BeanUtils会自动帮我们进行数据类型的转换。
BeanUtils.copyProperty(Object bean, String name, Object value);
BeanUtils.setProperty(Object bean, String name, Object value)
其中bean是指你将要设置的对象,name指的是将要设置的属性(写成"属性名"),value(值)
ConvertUtils.register(Converter converter, Class<?> clazz);
当需要将String数据转换成引用数据类型(自定义数据类型时),需要使用此方法实现转换。
BeanUtils.populate(Object bean, Map<String, ? extends Object> properties);
其中Map中的key必须与目标对象中的属性名相同,否则不能实现拷贝。
BeanUtils.copyProperties(newObject,oldObject);
实现对象的拷贝(谨慎使用这个copyproperties这个功能,相同的属性都会被替换,不管是否有值)
将对象转换为Map:
User user = new User();
Map userMap = BeanUtils.describe(user);
将Map转换为对象:
Map userMap = new HashMap();
User user = new User();
BeanUtils.populate(user,userMap);
BeanUtils convert转换
当用到BeanUtils的populate、copyProperties,getProperty,setProperty等方法其实都会调用convert进行转换:
但Converter只支持一些基本的类型,甚至连java.util.Date类型也不支持。当遇到不认识的类型时,会抛出异常来。
String[] old = {"12","2","3","5"};
long[] new = (long[])ConvertUtils.convert(old, long.class); //将String数组转换成long数组
int val = ConvertUtils.convert("2343",int.class); //字符串转int
这个时候就需要给类型注册转换器。比如:需要转成Date类型的数据都要通过DateLocaleConverter这个转换器的处理:
ConvertUtils.register(new DateLocaleConverter(), Date.class);
Person的birth变量是日期类型
Map map = new HashMap();
//自带转化器转化时间格式
map.put("birth", "2010-10-10");
ConvertUtils.register(new DateLocaleConverter(), Date.class);
Person p = new Person();
BeanUtils.populate(p, map);