Struts2配置精要之defaultStack与paramsPrepareParamsStack
在Struts2.3.4的struts-default.xml文件中配置了很多拦截器栈interceptor-stack:1.basicStack
2.validationWorkflowStack
3.fileUploadStack
4.modelDrivenStack
5.chainStack
6.i18nStack
7.paramsPrepareParamsStack
8.defaultStack
9.completeStack(这个跟defaultStack一模一样)
10.executeAndWaitStack
其中defaultStack与paramsPrepareParamsStack包含的interceptor最多,功能也最多,那么两者的区别是什么呢?
下面看看具体的顺序:
- <interceptor-stackname="paramsPrepareParamsStack">
- <interceptor-refname="exception"/>
- <interceptor-refname="alias"/>
- <interceptor-refname="i18n"/>
- <interceptor-refname="checkbox"/>
- <interceptor-refname="multiselect"/>
- <interceptor-refname="params">
- <paramname="excludeParams">dojo\..*,^struts\..*,^session\..*,^request\..*,^application\..*,^servlet(Request|Response)\..*,parameters\...*</param>
- </interceptor-ref>
- <interceptor-refname="servletConfig"/>
- <interceptor-refname="prepare"/>
- <interceptor-refname="chain"/>
- <interceptor-refname="modelDriven"/>
- <interceptor-refname="fileUpload"/>
- <interceptor-refname="staticParams"/>
- <interceptor-refname="actionMappingParams"/>
- <interceptor-refname="params">
- <paramname="excludeParams">dojo\..*,^struts\..*,^session\..*,^request\..*,^application\..*,^servlet(Request|Response)\..*,parameters\...*</param>
- </interceptor-ref>
- <interceptor-refname="conversionError"/>
- <interceptor-refname="validation">
- <paramname="excludeMethods">input,back,cancel,browse</param>
- </interceptor-ref>
- <interceptor-refname="workflow">
- <paramname="excludeMethods">input,back,cancel,browse</param>
- </interceptor-ref>
- </interceptor-stack>
顾名思义,这里的paramsPrepareParamsStack调用interceptor顺序是先params然后prepare最后再params,而defaultStack是直接prepare然后params(参考上一篇博文贴出的代码中的AbstractBaseAction),如果是defaultStack那么在prepareLoad()调用之前id是取不到页面传过来的值的,那么如果我加载一个人员信息页面的话,返回的页面字段都会是空值,如果用paramsPrepareParamsStack拦截器那么在prepareLoad()调用之前id就会被赋值之后执行空的load()方法返回视图页面就会取到值栈中的属性值。
- publicvoidprepareLoad()throwsException{
- if(getId()!=null){
- entity=getEntityById(getId());
- }
- }
- publicStringload()throwsException{
- returnSUCCESS;
- }
当使用struts2标签遇到这种错误的时候,你也应该考虑拦截器顺序的问题了:
- <s:checkboxlistlist="roles"listKey="id"listValue="name"name="selectRoleId"/>
- org.apache.jasper.JasperException:tag'checkboxlist',field'list',name'selectRoleId':Therequestedlistkey'roles'couldnotberesolvedasacollection/array/map/enumeration/iteratortype.
出现这个错误的原因可能如下:
1 刚进入该界面的时候发生错误,原因是 list="roles"中的这个集合是空的,导致错误
解决办法很简单,不能让list为空
2 刚进入该界面的时候list是有数据的,当点击提交等按钮的时候,数据被提交到后台,这时候在执行指定方法之前会调用拦截器,如果配置了验证框架或者在action中写了validate方法,校验没有通过,未走action,直接返回了input,又指定回了当前界面。
此时的checkboxlist中的list就没有初始化,值为空,一开始没有出错是因为执行了action里面对应的方法,而这里是拦截器validation拦截出错后直接跳转,没有执行action中应该执行的方法,导致了如上错误(这个错误提示的不太友好,让人认为是类转换错误)
解决办法是把初始化list的工作放到prepare拦截器中,因为prepare是在validate拦截器之前执行,所有默认的拦截器栈都是这样的顺序,即实现prepareble接口:
- publicclassRoleActionextendsActionSupportimplementsPreparable{
- @Override
- publicvoidprepare()throwsException{
- //初始化list
- }
- }
但是不推荐上面这种方法,最好是用get方法取值,页面加载的时候才取值。
- publicclassRoleActionextendsActionSupportimplementsPreparable{
- publicListgetRoles(){
- ....
- returnlist;
- }
- }