在web.xml中启用Struts2核心过滤器时,大家基本都按照这种写法:
<filter> <filter-name>struts2</filter-name> <filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class> </filter> <filter-mapping> <filter-name>struts2</filter-name> <url-pattern>/*</url-pattern> </filter-mapping>
这表明针对项目的上下文,会将所有的提交请求都转入到Struts2的处理,而且由于它是一个Filter,因此针对同一个请求会比别的Servlet要优先处理。
(注意,Filter的dispatcher默认为REQUEST(提交请求),重定向也算这种请求。如果Filter添加FORWARD的dispatcher,则转发请求也包含进来)
那么问题就来了:
1. 虽然所有请求都转入到Struts2处理,但为什么“只有”以.action结尾的请求才要进行Action映射,而像*.jsp, *.html,*.css, *.js等请求不用而且也不需要进行Action映射就能正常访问?
2. 为什么对于Servlet,*?wsdl等请求则要求Action映射,否则会Struts2会报“Could not find action or result...”的错误?
3. 为何在web.xml中的把Struts2的url-pattern改成"*.action"后,jsp中的Struts2标签失效?
其实,在url-pattern="/*"下,Struts2确实会把所有请求拦下来,不管是*.action,*.jsp, *.html,*.css, *.js还是Servlet,*?wsdl,这意味着这些被拦截下来的请求,
1. 如果里头使用Struts2的标签,则可以由Struts2解析出来。(这解释了问题3,如果*.jsp不由Struts2控制,那当然也无法使用Struts2的标签)
2. 有可能需要进行Action映射。
那到底什么请求要进行Action映射呢?在org.apache.struts2包下有一个default.properties,里头有一段默认配置:
### Used by the DefaultActionMapper ### You may provide a comma separated list, e.g. struts.action.extension=action,jnlp,do <span style="color:#FF0000;">### The blank extension allows you to match directory listings as well as pure action names ### without interfering with static resources, which can be specified as an empty string ### prior to a comma</span> e.g. struts.action.extension=, or struts.action.extension=x,y,z,, struts.action.extension=action,,
原来Struts2判断何种拦截下来的请求要进行Action映射正是通过这个配置指定的,默认包括扩展名为".action"的请求以及“空扩展名”的请求。
由此可见,尽管*.jsp, *.html,*.css, *.js被拦截下来了,但它们都有各自的非".action"扩展名,所以不满足参与Action映射的要求,因此也就能正常访问了。
而至于Servlet,*?wsdl等请求,它们则属于“空扩展名”,所以Struts2要求它们要参与Action映射,否则会报“Could not find action or result...”的错误。
另外,在开发Struts2时,有时会出现“循环转发”或“循环重定向”的错误,如以下错误配置:
web.xml:
<filter> <filter-name>struts2</filter-name> <filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class> </filter> <filter-mapping> <filter-name>struts2</filter-name> <url-pattern>/*</url-pattern> <dispatcher>REQUEST</dispatcher> <span style="color:#FF0000;"><dispatcher>FORWARD</dispatcher></span> </filter-mapping>
struts.xml
<package name="defaultMapping"> <action name="*"> <result>{1}</result><!--发生循环转发--> <!-- 发生循环重定向 <result type="redirect">{1}</result> --> </action> </package>
如果某个提交请求需要类似上面的默认转发(实现Action的放行),但由于在web.xml中Struts2的核心过滤器连转发请求也拦截了,所以Struts2又重新捕获到这个由Struts2自己转发出去的请求,然后又映射并转发出去,又再次捕获……从而造成循环转发。(循环重定向的过程类似于循环转发,不再多述)
版权声明:本文为博主原创文章,未经博主允许不得转载。