浅谈Spring的属性编辑器的使用
在 Spring 配置文件中,我们往往通过字面值设置 Bean 各种类型的属性值 ,这个功能是通过属性编辑器实现的。
创新互联公司制作网站网页找三站合一网站制作公司,专注于网页设计,成都网站建设、做网站,网站设计,企业网站搭建,网站开发,建网站业务,680元做网站,已为1000多家服务,创新互联公司网站建设将一如既往的为我们的客户提供最优质的网站建设、网络营销推广服务!
任何实现了 java.beans.PropertyEditor 接口的类都是属性编辑器 。 它可以将外部需要设置的值转换为 JVM 内部的对应类型,所以属性编辑器其实就是一个类型转换器 。
1 JavaBean 编辑器
Sun 所制定的 JavaBean 编辑器,很大程度上是为 IDE 准备的。它让 IDE 能够以可视化的方式来设置 JavaBean 的属性 。
Java 通过 java.beans.PropertyEditor 定义了设置 JavaBean 属性的方法,通过 BeanInfo 描述了 JavaBean 哪些属性是可定制的,此外还描述了可定制属性与 PropertyEditor 之间的对应关系 。
BeanInfo 与 JavaBean 之间的对应关系,通过两者之间规范的命名确立,对应 JavaBean 的 BeanInfo 采用如下命名规范:
Java 提供了一个用于管理默认属性编辑器的管理器: PropertyEditorManager ,它保存着一些常见类型的属性编辑器,如果某个 JavaBean 的常见类型属性没有通过 BeanInfo 显式指定它的属性编辑器, 那么 IDE 将自动使用 PropertyEditorManager 中注册的对应属性的默认编辑器 。
1.1 属性编辑器(PropertyEditor)
PropertyEditor 是属性编辑器接口,它定义了将外部设置值转换为内部 JavaBean 属性值的接口方法 。主要接口方法说明如下:
方法 | 说明 |
---|---|
Object getValue() | 返回属性的当前值 ,基本类型被封装成对应的包装类型 |
void setValue(Object newValue) | 设置属性的值,基本类型以包装类型传入 |
String getAsText() | 用字符串来表示属性对象,以便外部的属性编辑器能够以可视化的方式显示 。 默认返回 null ,表示该属性不能以字符串表示。 |
void setAsText(String text) | 用一个字符串去更新属性的内部值,它一般从外部属性编辑器传入的。 |
String[] getTags() | 返回表示有效属性值的字符串数组,以便属性编辑器能够以下拉框的方式进行展示 。 默认返回 null。 |
String getJavaInitializationString() | 为属性提供初始值,属性编辑器以此值作为属性的默认值 。 |
PropertyEditor 接口是内部属性值和外部设置值的沟通桥梁 。
Java 为 PropertyEditor 提供了一个方便的实现类: PropertyEditorSupport ,该类实现了 PropertyEditor 接口,我们可以通过扩展这个类来设计自己的属性编辑器 。
1.2 Bean 属性描述(BeanInfo)
BeanInfo 描述了 JavaBean 中的可编辑属性以及对应的属性编辑器,每一个属性对应一个属性描述器 PropertyDescriptor。
PropertyDescriptor 的构造函数有两个入参: PropertyDescriptor(String propertyName, Class beanClass) ,其中 propertyName 为属性名; beanClass 是 JavaBean 所对应的 Class。
PropertyDescriptor 还有一个 setPropertyEditorClass(Class propertyEditorClass) 方法,它可以为 JavaBean 属性指定编辑器 。
BeanInfo 接口中最重要的方法是:PropertyDescriptor[] getPropertyDescriptors() ,它会返回 JavaBean 的属性描述器数组 。
BeanInfo 接口的一个常用的实现类是 SimpleBeanInfo ,我们可以通过扩展这个类来实现自定义的功能 。
2 Spring 默认属性编辑器
Spring 的属性编辑器与传统的用于 IDE 开发的属性编辑器不同,它没有 UI 界面,只是将配置文件中的文本配置值转换为 Bean 属性的对应值 。
Spring 在 PropertyEditorRegistrySupport 中为常见的属性类型提供了默认属性编辑器,分为 3 大类,共有 32 个:
类型 | 说明 |
---|---|
基础数据类型 | 【1】基本数据类型,如: boolean、int 等; 【2】基本数据类型封装类,如: Boolean、Integer 等; 【3】基本数据类型数组: char[] 和 byte[] ; 【4】大数: BigDecimal 和 BigInteger 。 |
集合类 | Collection、Set、SortedSet、List 和 SortedMap。 |
资源类 | Class、Class[]、File、InputStream、Locale、Properties、Resource[] 和 URL。 |
PropertyEditorRegistrySupport 中有两个用于保存属性编辑器的 Map 类型变量:
变量名 | 说明 |
---|---|
defaultEditors | 保存默认属性类型的编辑器,元素的键为属性类型,值为对应的属性编辑器实例。 |
customEditors | 保存用户自定义的属性编辑器,元素的键值和 defaultEditors 相同 。 |
3 自定义 Spring 属性编辑器
如果我们的应用定义了特殊类型的属性,并且希望在配置文件中以字面值方式来配置属性值,那么就可以编写自定义属性编辑器并注册到 Spring 容器的方式来实现。
Spring 默认的属性编辑器大都扩展自 java.beans.PropertyEditorSupport,我们可以通过扩展 PropertyEditorSupport 来自定义属性编辑器 。在Spring 环境下仅需要将配置文件中字面值转换为属性类型的对象即可,并不需要提供 UI 界面,所以仅需要覆盖 PropertyEditorSupport 的 setAsText() 方法就可以啦 (∩_∩)O哈哈~。
假设我们有两个实体 Book 和 Author,希望在配置 Book 时,可以直接设置 Author 的名字。
Book.java
public class Book { /** * 作者 */ private Author author; /** * 书名 */ private String name; //省略 get/setter 方法 }
Author.java
public class Author { private String name; //省略 get/setter 方法 }
首先,自定义 author 的属性编辑器:
public class CustomPropertyEditor extends PropertyEditorSupport { @Override public void setAsText(String text) throws IllegalArgumentException { if(text==null||text.length()==0){ throw new IllegalArgumentException("格式错误"); } Author author=new Author(); author.setName(text); //调用父类的方法,来设置属性对象 setValue(author); } }
如果使用 BeanFactory ,则需要手工调用 registerCustomEditor(Class requiredType, PropertyEditor propertyEditor)
方法注册自定义属性编辑器;如果使用的是 ApplicationContext ,那么只需要在配置文件中注册 CustomEditorConfigurer 即可 。CustomEditorConfigurer 实现了BeanFactoryPostProcessor 接口,所以它是一个 Bean 的工厂后处理器 。
现在注册自定义的属性编辑器:
BeanWrapper 在设置 book 的 author 属性时,将检索自定义属性编辑器注册表,当发现 author 属性类型所对应的属性编辑器 CustomPropertyEditor 时,它就会这个定制的属性编辑器把 "村上春树" 转换为 Author 对象 。
按照规范, Java 会在 JavaBean 的相同类包下查找是否存在
的类;如果存在,就会自动使用
作为该 JavaBean 的属性编辑器 。Spring 也支持这个规范。
所以如果在类包下有一个名为 AuthorEditor 属性编辑器类,那么就无须在配置文件中注册自定义的属性编辑器啦O(∩_∩)O哈哈~
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持创新互联。
文章名称:浅谈Spring的属性编辑器的使用
链接URL:http://hbruida.cn/article/ipscip.html