您的位置:首页 > 编程语言 > Java开发

关于springMVC无法加载静态资源的问题

2017-08-27 13:21 681 查看


关于springMVC无法加载静态资源的问题

如何才能让springMVC不去拦截静态资源呢?首先先得了解下“/”与“/*”的区别。

我们大家都知道在使用spring时候需要在web.xml中配置以下代码:
<!-- 启动Web容器时,自动装配ApplicationContext的配置信息 -->
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener
</listener-class>
</listener>

<!-- springMVC监听器 -->
<servlet>
<servlet-name>SpringMVC</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet
</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:spring/spring-mvc.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>SpringMVC</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
在上面的url-parttern中配置的是/当然也可以配置/*,那么有什么区别呢?
如果使用/*,请求时可以通过DispatcherServlet转发到相应的Controller中的,但是返回的内容中如果是jsp还是会再次被拦截,这样导致404错误。
其实Spring 的Servlet拦截器匹配规则都可以自己定义
例如:

@Controller
@RequestMapping("/homePage")
public class IndexController {
@RequestMapping("/index")
public String toIndex(){
return "index";
}
}
1
2
3
4
5
6
7
8
1
2
3
4
5
6
7
8
1、拦截*.do、*.htm, 例如:/homePage/index.do
这是最传统的方式,最简单也最实用。不会导致静态文件(jpg,js,css)被拦截。
2、拦截/,例如:/user/add
可以实现现在很流行的REST风格。很多互联网类型的应用很喜欢这种风格的URL。
弊端:会导致静态文件(jpg,js,css)被拦截后不能正常显示。想实现REST风格,事情就是麻烦       一些。后面有解决办法还算简单。
3、拦截/*,这是一个错误的方式,请求可以走到Action中,但转到jsp时再次被拦截,不能访问到jsp.


**


如何访问到静态的文件,如jpg,js,css?

如果你的DispatcherServlet拦截"*.do"这样的有后缀的URL,就不存在访问不到静态资源的问题。
但是如果你的DispatcherServlet拦截"/",为了实现REST风格,拦截了所有的请求,那么同时对*.js,*.jpg等静态文件的访问也就被拦截了。


  目的:可以正常访问静态文件,不可以找不到静态文件报404。


方案一:Tomcat的defaultServlet来处理静态文件

<servlet-mapping>
<servlet-name>default</servlet-name>
<url-pattern>*.css</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>default</servlet-name>
<url-pattern>*.js</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>default</servlet-name>
<url-pattern>*.gif</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>default</servlet-name>
<url-pattern>*.png</url-pattern>
</servlet-mapping>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16

  忧缺点:1. 要配置多个,每种文件配置一个。2. 要写在DispatcherServlet的前面, 让 defaultServlet先拦截请求,这样请求就不会进入Spring了。3. 高性能。


备注:

Tomcat, Jetty, JBoss, and GlassFish 自带的默认Servlet的名字 -- "default"
Google App Engine 自带的 默认Servlet的名字 -- "_ah_default"
Resin 自带的 默认Servlet的名字 -- "resin-file"
WebLogic 自带的 默认Servlet的名字  -- "FileServlet"
WebSphere  自带的 默认Servlet的名字 -- "SimpleFileServlet"


方案二:使用mvc:resources (spring3.0.4版本之后)

   对静态资源文件的访问 

   

  images/**映射到 ResourceHttpRequestHandler进行处理,location指定静态资源的位置.可以是web application根目录下、jar包里面,这样可以把静态资源压缩到jar包中cache-period 可以使得静态资源进行web cache 

  如果出现下面的错误, 



org.springframework.web.servlet.DispatcherServlet.noHandlerFound(1136) | No mapping found for HTTP request with URI [/BaoSteelCenter/loginPage/tologin] in DispatcherServlet with name 'SpringMVC'
1
1

可能是没有配置
<mvc:annotation-driven />
的原因。 

使用
<mvc:resources/>
元素,把mapping的URI注册到SimpleUrlHandlerMapping的urlMap中,key为mapping的URI pattern值,而value为ResourceHttpRequestHandler, 

  这样就巧妙的把对静态资源的访问由HandlerMapping转到ResourceHttpRequestHandler处理并返回,所以就支持classpath目录,jar包内静态资源的访问. 

  另外需要注意的一点是,不要对SimpleUrlHandlerMapping设置defaultHandler.因为对static URI的defaultHandler就是ResourceHttpRequestHandler, 

  否则无法处理static resources request.


方案三 ,使用
<mvc:default-servlet-handler/>

把”/**” url,注册到SimpleUrlHandlerMapping的urlMap中,把对静态资源的访问由HandlerMapping转到 org.springframework.web.servlet.resource.DefaultServletHttpRequestHandler 处理并返回.DefaultServletHttpRequestHandler使用就是各个Servlet容器自己的默认Servlet.


补充说明:多个HandlerMapping的执行顺序问题:

  DefaultAnnotationHandlerMapping的order属性值是:0

  < mvc:resources/ > 自动注册的 SimpleUrlHandlerMapping 的order属性值是: 2147483646

  
<mvc:default-servlet-handler/>
自动注册 的SimpleUrlHandlerMapping 的order属性值是: 2147483647

  spring会先执行order值比较小的。当访问一个a.jpg图片文件时,先通过 DefaultAnnotationHandlerMapping 来找处理器,一定是找不到的,因为我们没有叫a.jpg的Action。然后再按order值升序找,由于最后一个 SimpleUrlHandlerMapping 是匹配 “/**”的,所以一定会匹配上,就可以响应图片。 访问一个图片,还要走层层匹配。不知性能如何?

  最后再说明一下,方案二、方案三 在访问静态资源时,如果有匹配的(近似)总拦截器,就会走拦截器。如果你在拦截中实现权限检查,要注意过滤这些对静态文件的请求。

  如何你的DispatcherServlet拦截 *.do这样的URL后缀,就不存上述问题了。还是有后缀方便。

以上内容参考[网络内容]感谢作者,本人略加整理。

转自:原文链接
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息