一、首先需要明确几容易混淆的规则:
- servlet容器中的匹配规则既不是简单的通配,也不是正则表达式,而是特定的规则。
- Servlet 2.5开始,一个servlet可以使用多个url-pattern规则,<servlet-mapping>标签声明了与该servlet相应的匹配规则,每个<url-pattern>标签代表1个匹配规则;
- 当servlet容器接收到浏览器发起的一个url请求后,容器会用url减去当前应用的上下文路径,以剩余的字符串作为servlet映射,假如url是http://localhost:8080/ServletDemo/index.html,其应用上下文是ServletDemo,容器会将http://localhost:8080/ServletDemo去掉,用剩下的/index.html部分拿来做servlet的映射匹配(filter不同)
- url-pattern映射匹配过程是有优先顺序的(精确匹配>路径匹配>扩展名匹配>缺省匹配)
- 而且当有一个servlet匹配成功以后,就不会去访问剩下的servlet了。
二、四种匹配规则
1、精确匹配
<url-pattern>中配置的项必须与url完全精确匹配。
<servlet-mapping>
<servlet-name>ServletDemo</servlet-name>
<url-pattern>/login/login.html</url-pattern>
<url-pattern>/index.html</url-pattern>
<url-pattern>/login/login.action</url-pattern>
</servlet-mapping>
当在浏览器中输入如下几种url时,都会被匹配到该servlet
http://localhost:8080/ServletDemo/login/login.html
http://localhost:8080/ServletDemo/index.html
http://localhost:8080/ServletDemo/login/login.action
但是http://localhost:8080/ServletDemo/login/login/ 是非法的url,不会被识别
url后面可以跟任意的查询条件,都会被匹配,如
http://localhost:8080/ServletDemo/login/login.action?username=admin&age=admin。
2 路径匹配
以“/”字符开头,并以“/*”结尾的字符串用于路径匹配
<servlet-mapping>
<servlet-name>ServletDemo</servlet-name>
<url-pattern>/login/*</url-pattern>
</servlet-mapping>
路径以/login/开始,后面的路径可以任意。比如下面的url都会被匹配。
http://localhost:8080/ServletDemo/login/login.html
http://localhost:8080/ServletDemo/login/login.action
http://localhost:8080/ServletDemo/login/register.action
3 扩展名匹配
以“*.”开头的字符串被用于扩展名匹配
<servlet-mapping>
<servlet-name>ServletDemo</servlet-name>
<url-pattern>*.jsp</url-pattern>
<url-pattern>*.action</url-pattern>
</servlet-mapping>
任何扩展名为jsp或action的url请求都会匹配,比如下面的url都会被匹配
http://localhost:8080/ServletDemo/login/login.jsp
http://localhost:8080/ServletDemo/login.action
4 缺省匹配
<servlet-mapping>
<servlet-name>ServletDemo</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
这种往往会与“/*”概念弄混淆,这种表式过滤所有请求,但是不过滤jsp、asp等动态扩展名请求
.注意 如果前面三条规则都没有找到一个servlet,容器会根据url选择对应的请求资源。如果应用定义了一个default servlet,则容器会将请求丢给default servlet
三、匹配顺序
1、精确匹配
<servlet-mapping>
<servlet-name>ServletDemo</servlet-name>
<url-pattern>/login/login.html</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>ServletDemo</servlet-name>
<url-pattern>/*</url-pattern>
</servlet-mapping>
当一个请求http://localhost:8080/ServletDemo/login/login.html来的时候,会首先匹配第一个<servlet-mapping>,不再匹配第二个 <servlet-mapping>
2、路径匹配
<servlet-mapping>
<servlet-name>ServletDemo</servlet-name>
<url-pattern>/page/login/login.html</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>ServletDemo</servlet-name>
<url-pattern>/page/*</url-pattern>
</servlet-mapping>
先匹配最长路径,再匹配最短路径。当一个请求http://localhost:8080/ServletDemo/page/login/login.html来的时候,会首先匹配第一个<servlet-mapping>,不再匹配第二个 <servlet-mapping>
3扩展名匹配
<servlet-mapping>
<servlet-name>ServletDemo</servlet-name>
<url-pattern>/login/*</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>ServletDemo</servlet-name>
<url-pattern>*.action</url-pattern>
</servlet-mapping>
当一个请求http://localhost:8080/ServletDemo/login/login.action来的时候,会首先匹配第一个<servlet-mapping>,不再匹配第二个 <servlet-mapping>
4、缺省匹配:
以上都找不到servlet,就用默认的servlet,配置为<url-pattern>/</url-pattern>
四、特别注意
1、路径匹配和扩展名匹配无法同时设置
匹配方法只有三种,要么是路径匹配(以“/”字符开头,并以“/*”结尾),要么是扩展名匹配(以“*.”开头),要么是精确匹配,三种匹配方法不能进行组合,不要想当然使用通配符或正则规则。
如<url-pattern>/user/*.action</url-pattern>是非法的
另外注意:<url-pattern>/aa/*/bb</url-pattern>是精确匹配,合法,这里的*不是通配的含义
2、在web.xml两种特殊格式
第一种:
<servlet-mapping>
<servlet-name>ServletDemo</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
第二种:
<servlet-mapping>
<servlet-name>ServletDemo</servlet-name>
<url-pattern>/*</url-pattern>
</servlet-mapping>
- “/*”属于路径匹配,并且可以匹配所有request,由于路径匹配的优先级仅次于精确匹配,所以“/*”会覆盖所有的扩展名匹配,很多404错误均由此引起,所以这是一种特别恶劣的匹配模式,一般只用于filter的url-pattern
- “/”是servlet中特殊的匹配模式,切该模式有且仅有一个实例,优先级最低,不会覆盖其他任何url-pattern,只是会替换servlet容器的内建default servlet ,该模式同样会匹配所有request。
- 配置“/”后,一种可能的现象是ServletDemo会拦截诸如http://localhost:8080/ServletDemo/user/addUser.action、http://localhost:8080/ServletDemo/login/login的格式的请求,但是并不会拦截http://localhost:8080/ServletDemo/login/login.jsp、http://localhost:8080/ServletDemo/index.jsp,这是应为servlet容器有内置的“*.jsp”匹配器,而扩展名匹配的优先级高于缺省匹配,所以才会有上述现象。
-
注:为避免静态资源被拦截需要在spring.xml文件中配置
-
<mvc:resources location="/images/" mapping="/images/**"/> <mvc:resources location="/css/" mapping="/css/**"/> <mvc:resources location="/js/" mapping="/js/**"/>
/js/*代表的是js文件夹下的所有子文件,/js/**代表的是js文件夹下的所有子文件及子文件夹中的子文件 ,
当你的请求路径是http://localhost:8080/SpringMVCDemo/demo2/js/j.js会访问js文件夹下的j.js文件,如果是http://localhost:8080/SpringMVCDemo/demo2/js/js/j.js会访问js文件夹下的js文件夹的j.js文件,这只是为了解释mapping配置*与**的关系
-
关于location与mapping之间的关系
-
mapping相当于请求格式,而location是资源的位置
如果把js文件夹放到WEB-INF中
那应该修改mapping还是location?
这应该就明白了