综述:字符串到 Java 对象转换的工具库

  • Tim Cull
  • 张凯峰

2010 年 9 月 19 日

话题:Java语言 & 开发架构

Joda Time fame的 Stephen Colebourne关于他之前发布的一个 Java 库Joda Convert 在他的 blog 上率先发起了一场小范围的争论这个库可以通过 annotation 来进行基础对象和字符串之间的转换。为了解释清楚关于字符串转换的方方面面,下面是对字符串和对象间互相转换技术的一个综述,让我们从 Joda Convert 开始。

根据 Colebourne 先生的文章,Joda Convert 宣称的目标是为了简单性而牺牲完整性。它允许 Java 类的作者指定任意名称的方法,实现从字符串到此类实例间的转换。比 如,Currency 类可能有个静态方法叫做“fromISOCode(code)”,以及一个实例方法“getISOCode()”。如果为这两个方法 添加来自 Joda Convert 的 annotation:@FromString 和 @ToString,使用 Currency 类的应用就可以通过类似这样的调用:“Foo bar = StringConvert.INSTANCE.convertFromString(Foo.class, str);”,来实现对象和字符串间的转换。这样的转换通常对那些必须解析来自 HTTP GET 数据的 Web 应用很有帮助。Colebourne 先生的这篇 blog 的评论者们则提出了一些其他的可替代方案。

第一个可替代方案是 Java 中内嵌的 java.beans.PropertyEditor。PropertyEditor 使用 JavaBeans 技术来把字符串转换成属性值。虽然 PropertyEditor一般常见于像 IDE 这样的可视化编辑器里面,但其实它也可以用在后台的处理上。比如Spring3.0 之前的版本广 泛地使用了 PropertyEditor 来支持数据绑定和验证。不像 Joda Convert,PropertyEditor 可以用在很多方面,而不仅仅是把字符串转换成对象。比如,PropertyEditor 有内建的对注册 PropertyChangeListener 的支持。

J2EE 在JSF中也有自己内建的转换技术。JSF 中包含了用于普通类型比如 BigDecimal、Float、DateTime 等的转换器,也提供了一个接口用于自定义转换器实现。自定义实现只需要为 getAsObject 和 getAsString 方法提供代码。但是 JSF 转换器和 JSF 绑定得很紧,所以在 JSF 以外使用它们会很困难。还有一些其他第三方的替代方案实现了字符串转换技术。其中一个最近被用在 Spring 3.0 中。虽然 Spring 之前的版本使用 PropertyEditor,但 SpringSource 选择在 3.0 里面实现自己的转换方法。根据 SpringSource 所说:

当我们开始改善 Spring 3 的数据绑定系统时,我们的目标是:

1. 提供一个无状态、强类型的类型转换 SPI 来取代 JavaBean PropertyEditor

2. 提供一个统一的类型转换 API,用在任何需要的地方,包括 Spring 的 DataBinder 和 Expression Language

3. 可以通过 Java 注解元数据来驱动类型转换

4. 通过注册可感知的缺省设置,以及采用惯例优于配置的策略,来简化转换

SpringSource 并不是唯一一家提供自己的类型转换器的开源组织。一些 Apache 项目也有他们自己的类型转换器,比如Apache Commons BeanUtils(用在Apache Digester),陷入停滞的Apache Commons Convert,甚至是Apache StrutsApache Camel

一些框架还被设计成不仅仅能把字符串转换成对象。比如,Dozer 是一个可以把任意复杂的对象转换成其他任意复杂的对象。因为字符串就是对象,所以它可以处在 Dozer 转换的任意一方。

最后,在选择一种转换技术的时候,开发者需要记住的是,上面提到的任何一种技术都是针对自己的目标环境而优化的。比如,浏览器中发送到 HTML 表单以及接收 自 HTML 表单的字符串,就会和把同一个对象写成 XML 文档的字符串不太一样。此外,即使是在完全相同的目标环境,比如浏览器中,一些用户可能需要的是不 同的本地化字符串(比如日期“2010-31-01”或者“1/31/2010”)。最后但一样重要的是,不同的媒介会支持不同的字符串编码。比如,XML 文档可以使用 UTF-8、ISO-8859-1或者许多其他的编码,因为它可以在第一行中指定字符串编码。而从 URL 解析得到的字符串只能是ASCII编码。



查看英文原文:Roundup of String to Java Object Conversion Libraries

Java语言 & 开发架构