Struts2第二天学习
解决struts.xml文件的冗余
将所有的配置文件放在一份struts.xml将会导致, 文件繁杂且容易出错, 将有如下的解决方案:
分文件编写Struts2的配置文件:对于具有类似功能的action就可以放入不同对应的xml文件中, 最后使用struts标签中的<include file="user.xml"></include>处理例如:1. struts.xml<struts><constant name="struts.devMode" value="true"></constant><include file="A.xml"></include><include file="B.xml"></include></struts>2. A.xml<struts><package name="A" extends="struts-default"><action name="addA"><result>/success.jsp</result></action></package> </struts>3. B.xml<struts><package name="B" extends="struts-default"><action name="addB"><result>/success.jsp</result></action></package> </struts>
封装请求数据到对象中
- 静态参数封装: 使用struts.xml配置param参数, 调用动作类的setXXX方法做数据注入,默认编码utf-8, 不会出现乱码问题
例: struts.xml配置<struts><package name="user" extends="struts-default"><action name="addUser" class="com.action.UserAction" method="saveUser"><param name="username">A</param><param name="age">18</param></action></package> </struts>在动作类中的配置:public class UserAction extends ActionSupport{private String username;private int age;public String getUsername(){return username;}public void setUsername(String username){this.username = username;}public int getAge(){return age;}public int setAge(){this.age = age;}public String saveUser(){ //action方法return null; } }
当访问index.jsp中addUser操作的时候, struts.xml触发saveUser的action, 将username, age中参数通过UserAction中的set方法做注入操作
整个注入的过程中, 首先会自动为UserAction生成一个user对象, 将username, age通过set方式进行赋值
Struts2是通过staticParams的拦截器做注入操作, 这里注意必须保证, 注入的param标签中的name必须与Action类中的属性同名
在param中的age参数18是作为字符串出现的, 当使用set操作的时候, 字符串自动转换为数字
- 动态参数封装: 就是对用户提交的表单数据封装到实体模型中
当用户使用表单提交数据的时候, 在Servlet中, 直接使用request.getParameter("参数名")处理
而在Struts2中, 系统默认将表单数据一一通过set操作封装到Action类的属性中
在struts.xml不需要做参数配置, 例如:<action name="addUser" class="com.action.UserAction" method="saveUser"></action>无需指定param
注: 此处的操作是由params拦截器完成的, 通过拦截器将表单数据填入对应的Action类中同名同类型使用此种方式出现的问题是: Action不应该处理对JavaBean数据的处理, Action的主要功能是对用户操作的反馈针对上述的问题, 提出如下的操作
1. 动作类与实体模型分开
在动作类中不采用上面的方式(将username, age等属性放入动作类), 而是将数据单独封装为User类处理
例:public class UserAction extends ActionSupport{private User user;public String getUser(){return user;}public void setUser(User use){this.user = user;}public String saveUser(){ //action方法return null; } }
在index中, 作如下处理
用户名: <input name="text" name="user.username"/>
此处说明: user.username就类似于Servlet中的将数据存入request, session中传递
注: 在index.jsp中要注意: user.username中, user必须与Action中属性名相同, username必须与User中的属性名相同在表单数据封装到实体类中分为以下2个步骤:1. 通过Action类中的get方法查看当前User实例是否存在, 不存在则new一个, 然后使用Action中的set方法为bean引用赋值2. 调用action中get方法获得bean对象, 通过注入的方式为实例对象赋值, 整个过程中在struts.xml无需做配置, 过滤器会自动将表单中数据封装到action中对象上
此时的赋值一样是param拦截器进行处理
需要注意的, 当在上述的1操作中, 如果bean的实例对象存在, 那么就不会执行set操作, 直接执行get操作, 最后对该实例对象赋值
2. 在动作类与实体模型分开的基础上, 进行模型驱动
如果只是单纯地将动作类与实体模型分开, 容易出现的问题就是在写jsp的过程中必须保证, 模型变量之间的联系, 如果某个地方的变量名错误, 很容易导致错误
当采用模型驱动的方式, 在一定程度上可以减轻jsp的开发
如标题所说, 模型驱动就是使用已存在的bean实例驱使jsp中表单数据进行封装, 而不是让jsp去指定向哪个实例中的哪个属性进行封装数据
模型驱动的2个步骤:
1. 让Action实现ModelDriven接口, 实现getModel方法(返回值泛型)
2. 使用模型驱动, 数据模型必须自己实例化
例:Action类: public class UserAction extends ActionSupport implements ModelDriven<User>{private User user = new User();public User getModel(){//重写ModelDriven中的getModelreturn user;}public String registerUser(){return null; }}index.jsp:用户名:<input type="text" name="username"/><br/>年龄:<input type="text" name="age"/><br/><input type="submit" value="注册"/>struts.xml:<action name="registerUser" class="com.action.UserAction" method="registerUser"></action>
注: 模型驱动使用的是ModelDriven拦截器
数据类型转换
- 需要注意的是, 通过客户端填写的数据都是String或String[]
- 获取表单中字符串数据 -> 使用set方法填充模型数据 -> POJO- POJO -> 使用get方法获取数据 -> jsp显示(字符串类型)
注: 在set填充数据的时候Struts2会根据数据类型做自动转换处理
注: 系统的类型转换在某些方面可能不满足实际需求, 需要我们自己定义类型转换器.
-
下面介绍一下自定类型转换器
-
在Struts2中具有一个TypeConvert的接口, 当实现该接口就可完成自定义类型的转换, 存在的问题是该接口中的convertValue方法显得过于繁琐, 因此一般采用继承TypeConvert的子类解决问题
-
编写类继承TypeConvert的子类StrutsTypeConverter, 实现抽象方法: public abstract Object convertFromString(Map context, String[] values, Class toClass); 以及 public abstract String convertToString(Map context, Object o);
convertFromString是将用户输入的String数据转化为指定类型
convertToString是将指定该类型转化为String类型
context: 存放动作访问的所有数据
values: 存放用户输入的值
toClass: 要转换的目标类
以转换日期为例:
public class MyTypeConverter extends StrutsTypeConverter{private DateFormat df = new SimpleDateFormat("mm/dd/yyyy"); //例如将1/1/2019变为字符串, 按照当地时间格式存入数据库//重写convertFormStringpublic Object convertFromString(Map context, String[] values, Class toClass){if (values == null || values.length == 0){return null;}String str = values[0];if(toClass == java.util.Date.class){try{return df.parse(str);} catch(Exception e){e.printStackTrace() return null;}}}return null;
}
//重写convertToString
public abstract String convertToString(Map context, Object o){if(o instanceof Date){return df.format((Date)o);} return null;
}
- 注册转换器
- 局部注册: 在javaBean包下建立一个 javaBeanName-conversion.properties的文件javaBeanName代表类名, 文件名必须按照上面的格式在.properties文件中将需要转换的属性注册转换器例: birthday=com.convert..MyTypeConverter- 全局类型转换器: 需要使用转换的数据类型注册转换器在src路径下, 建立一个xwork-conversion.properties属性的文件, 这里的文件名也是固定的例: java.util.Date=com.convert..MyTypeConverter此处由于针对全局元素, 所以直接将Data类型作为转换器注册对象
经过上述的步骤就能将指定类型转换为本身需要的类型
转换失败后的处理
转换失败后不能将异常抛给用户, 所以, 一般采用如下操作:
在结果视图标签中, 添加<result name="input">/index.jsp</result>
当出现异常的时候, Struts2会自动切换到input视图
处理input回显数据
常规的Servlet回显数据使用的是将数据存入request中, 通过EL表达式在表单中回显数据
而Struts2采用的是配置标签, 使用Struts2自带标签, 在使用input回显操作的时候将会对数据做自动填充, 填充至表单中, Struts2中自带的标签就是对原本的html标签的一种修饰以及功能加强
如下操作:
<%@ taglib uri="/struts-tags" perfix="s"%> <%--使用Struts2自带标签--%>
<s:form><s:textfield name="username" label="用户名"></s:textfield><s:textfield name="password" label="密码"></s:password>
</s:form>
剩余标签以及内部属性,自行查阅
当使用Struts2提供的标签库的时候, 数据的回显, 不再需要"request封装"
表单信息提示
当用户提交表单数据的时候, 需要对表单数据做验证操作, 如果信息错误则给用户一个错误提示, 在Servlet中, 采用的是request封装数据, 结合js做验证
在Struts2中采用的也是标签配置, 例如:
<s:fielderror></s:fielderror> <%--字段错误提示, 此处的s借用上面的--%>
<s:actionerror/> <%--动作错误--%>
当字段出现错误的时候, 做数据回显, 会作自动的提示操作, 告诉用户对应的输入出现什么样的错误
注: 当出现数据类型转化错误, 将会触发conversionError的拦截器(前提必须是Action继承ActionSupport), 该拦截器将会进入input视图如果没有对字段信息做处理, 将会以默认的方式显示, 显示一堆英文对用户不友好, 所以下面将针对编写自定义信息提示做叙述
自定义表单数据信息提示以及验证规则
一般采用客户端与服务端共同对信息进行验证
下面介绍Struts2的服务端验证, 下面的所有的操作都是基于Action类继承ActionSupport来处理的
- 自定义表单数据信息提示
form表单中的数据总是针对JavaBean某个字段而言, 所以有以下处理:
在JavaBean路径下, 新建一个与Bean类名相同的properties的文件, 然后针对需要自定义的字段进行信息配置
例如:在User.properties中invalid.fieldvalue.birthday=出生日期有误, 使用yyyy-MM-dd格式注: 在properties文件中, 中文会自动用Unicode编码代替文件的key必须是invalid.fieldvalue.属性名文件value代表显示的中文提示信息
- 处理Action类中动作方法的验证
使用重写validate方法解决验证
在ActionSupport中提供了一个validate方法, 那是系统默认的验证操作, 自定义验证规则就需要重写父类方法
例:public void validate(){if (StringUtils.isEmpty(user.getUsername)){addFieldError("username", "用户名必须输入");} }
//在Struts2中提供一个Map用于封装操作错误信息, 通过addFieldError方法添加错误信息
//"username"代表错误信息的字段名, "用户名必须输入"代表错误提示信息
注: 使用上面的validate操作具有一个弊端, 就是会对所有的动作操作做验证规则, 如上面的操作, 如果user的username一直为空, 当用户执行其他动作操作, 就会报错
例如:
在Action类中具有一个findAll操作
public String findAll(){return SUCCESS;
}
当用户没有注册( 此时的username为null )而去执行findAll操作, 出现的问题就是: 即使findAll操作成功. 但是依旧抛异常
出现这种情况的原因就是: validate对所有的动作都进行验证
解决上述问题, 采用有如下的方案:
-
给不需要验证的动作添加一个@SkipValidation注解
如上面的操作, 为findAll添加一个@SkipValidation注解, 使得跳过findAll, 不去验证它 -
validation方法遵循书写规范
将上面的validate改为validateRegist, 使得用户注册的时候只验证regist动作, 而不去验证findAll动作
声明式验证, 不使用函数式验证方式, 而是编写xml配置文件, 制定验证规则
在动作类所在的目录下, 新建一个 动作类类名-validation.xml的文件
例如:
文件名: UserAction-validation.xml
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE validators PUBLIC"-//Apache Struts//XWork Validator 1.0.3//EN""http://struts.apache.org/dtds/xwork-validator-1.0.3.dtd"><validators><field name="username"><field-validator type="requiredString"><message>请输入用户名</message></field-validator></field> </validators>对username属性做requiredString验证(表示username不能为空)
注: 它是针对动作类中所有的动作方法做验证
在动作类所在的目录下, 新建一个 动作类类名-动作名-validation.xml的文件就可以实现对特定的动作进行验证
例如:
文件名: UserAction-regist-validation.xml
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE validators PUBLIC"-//Apache Struts//XWork Validator 1.0.3//EN""http://struts.apache.org/dtds/xwork-validator-1.0.3.dtd"><validators><field name="username"><field-validator type="requiredString"><message>请输入用户名</message></field-validator></field></validators>Struts.xml中代码<package name="user" extends="struts-default"><action name="findAll" class="com.action.UserAction"><result>/success.jsp</result></action><action name="regist" class="com.action.UserAction" method="regist"><result type="redirect">/success.jsp</result><result type="exists">/message.jsp</result><result type="input">/success.jsp</result> <%--视图回显--%></action></package>此时的操作就只是单纯针对于regist
- 验证器参数注入
只是单纯地配置validation.xml并不能满足对输入数据的检查要求, 它只能做到根据field标签中type类型来检查数据, 所以需要使用验证器参数注入增强验证功能
常用验证器自行百度
验证器参数注入有两种方式, 如下:
基于字段: 先获取需要验证的字段名, 然后对其内容进行验证
例如:
<validators><field name="username"><field-validator type="requiredString"><param name="trim">false</param><message>请输入用户名</message></field-validator></field></validators>上面针对type为requireString的验证器做trim注入, 将该验证器的trim属性设置为false表示验证器不除去空格
基于验证器: 先判断当前需要验证的类型属于哪一类, 然后再判断需要验证哪个字段
例如:
<validators type="requiredString"><validator><param name="fieldname">username</param><message>请输入用户名</message></validator></validators>对username做注入, 表示对username做判断requiredString操作
注: 一个验证器只能处理一个验证请求, 需要对同一属性做多种条件判断, 需要多个< field-validator>, 例如对password判断, 需要判断password长度, 以及password是否为已输入.
<validators><field name="password"><field-validator type="requiredString"><message>请输入密码</message></field-validator><field-validator type="stringlength"><param name="minlength">3</param><param name="maxlength">8</param><message>密码长度${minlength}~${maxlength}</message></field-validator></field></validators>
还需要注意的是, 密码中含有特殊字符, 可以将密码作为文本处理, 防止特殊字符出错