浏览器服务器之间的数据通信不可能有丰富的数据类型,实际上http参数都是字符串类型的。因此必须在服务器端完成数据类型的转换。Struts2有强大的内建转换机制,类型转换可以依赖于OGNL表达式。开发者也可以自己创建类型转换器。


内建的类型转换器和基于OGNL类型转换

Struts2可以完成大多数常用的类型转换。借助于OGNL,Struts2允许以另一种简单方式请求参数转换成符合类型。比如表单需要提交的action中有一个user类型属性,那么提交的时候直接使用user.name和user.pass即可将表单数据传给user实例的相应属性。这种方式能够将普通的http请求参数(字符串)转化为符合类型对象(user对象,而非普通的String,int等)。演示代码如下:

<s:form action="lalala/login">
<s:textfield name="user.name" label="U" />
<s:textfield name="user.gender" label="gender"/>
<s:textfield name="username" label="username"/>
<s:submit value="adg"/>
</s:form>

此表单会将username中内容赋给login中的username属性,将name与gender赋给login中user实例的name与gender属性。

<s:property value="user.gender"/><br/>
<s:property value="user.name"/><br/>

在success视图页,通过property标签提取相关属性值。

此外如果想要给login这个类的Map<String,user> users赋值,我们就既需要指明给哪个value赋值,也需要指明这个值是此value下的什么属性。例如:

<s:form action="lalala/login">
<s:textfield name="users['1'].gender" label="gender1" />
<s:textfield name="users['1'].name" label="name1" />
<s:textfield name="users['2'].gender" label="gender2" />
<s:textfield name="users['2'].name" label="name1" />

<s:submit value="submit"/>

jsp中访问的时候:

<s:property value="users['1'].gender"/>

--------------------------------------------------

如果把map换成list,写法一样,只不过[]中的内容不同。map的[]中的内容是value,是String类型。而list的[]是数字,是int类型。不过它们都是user对象的标识(id)。

不使用泛型的时候,架构是没办法使用类型转换器处理users属性的,因此需要一个局部类型转换文件。这个文件是与action相对应的。文件中不仅可以指定action中的某个属性使用哪个自定义类型转换器,还能够指定此action的list元素是哪种对象类型。在login-conversion.properties中填写如下一行代码:

Element_users=com.cm.actions.user

此种代码专门为list而生,users指定的就是list中的元素类型。


自定义类型转换器

如果用户输入的是aaa,bbb,而我们想让aaa成为action中user属性的username,bbb成为user属性的password,就需要使用自定义类型转换器。自定义类型转换器能够将一个输入的字符串预处理,按照自定义的方式转换成相应实例的某种属性类型。转换器必须实现TypeConverter接口。但是这个接口过于复杂,所以GONL提供了一个DefaultTypeConverter类,它是这个接口的实现类。自定义转换器要继承该类,然后重写convertValue方法。一个自定义转换器能够完成双向的类型转换。书中的例子是,User可以转成字符串,字符串也可以转成User类型。为了确定转换方向,需要判断toType属性,它代表的目标属性,究竟是想要转到String,还是User。此方法中的三个参数,context是类型转化上下文,当需要转换ActionContext的时候使用。value是需要转换的参数,它其实是一个字符串数组,toType是目标类型。参考文章actioncontext

上述方法是基于非框架的。Struts2框架还提供了一个StrutsTypeConverter抽象类,它更好地实现了TypeConverter接口,将两个转换方向分成两个不同的方法。代码如下:

public class userconverter extends StrutsTypeConverter {

	@Override
	public Object convertFromString(Map arg0, String[] arg1, Class arg2) {
		user user = new user();
		String[] userValues = arg1[0].split(",");
		user.setGender(userValues[1]);
		user.setName(userValues[0]);
		// TODO Auto-generated method stub
		return user;
	}

	@Override
	public String convertToString(Map arg0, Object arg1) {
		user user = (user) arg1;
		// TODO Auto-generated method stub
		return user.getName() + user.getGender();
	}

}

这里我们注意到,converFromString方法中的String是从一个字符串数组中取出来的。为什么不直接使用一个字符串呢?那是因为有些时候的输入,比如多选框,下拉框,传进来的String可能是很多个,它们对应的请求参数就是字符串数组。

由于现在user的两个属性值由同一个字符串表示了,所以在写表单的时候,不需要再用OGNL表达式来制定aciton的属性的属性了,直接写action的属性名。如下

<s:form action="lalala/login">
<s:textfield name="user" label="U and gender" />
<s:textfield name="username" label="username"/>
<s:submit value="submit"/>

注册类型转换器

我们需要告诉架构什么时候需要使用哪一个类型转换器,因此我们要将类型转换器注册。我们可以注册全局,局部类型转换器,也可以通过JDK的注解实现。

局部类型转换器使用局部类型转换文件指定。此文件应该与对应的action同一目录,严格按照命名规则,ActionName-conversion.properties。此文件中添加如下一行:

user=com.cm.converter.userconverter

局部转换器只对一个action起作用(因为是根据actionclassname匹配的)。全局类型转换器能够针对多个action的user属性,或者一个action中的多个user属性进行类型转换。在WEB-INF/classes下创建一个xwor-conversion.properties文件,里面进行注册。此时注册代码要把包名打全。

com.cm.actions.login.user=com.cm.converter.userconverter

错误处理

由于表现层的用户输入可能存在错误,因此有必要对数据的正确性进行校验,同时提供错误处理的方法。实际上struts2中的校验跟类型转换是紧密相关的。因为在类型转换前,系统必须确保数据已经经过校验。

在默认xml中我们发现有一个conversionError拦截器,该拦截器能够处理转换中的异常,将错误封装成表单错误FieldError,并将这些信息放在ActionContext中。在处理页面(result为input的页面)中使用s:fielderror标签即可输出错误信息。为了使用这套机制,action必须继承ActionSupport类。

如果想国际化地输出类型转换错误信息,则要在国际化资源文件中增加key为xwork.default.invalid.fieldvalue的字段。