为什么SpringMVC在访问静态资源的时候,对于html、js等静态资源访问不到?但jsp却可以?
原因
这个问题要先从<url-pattern>/<urlpattern>
和<url-pattern>/*<url-pattern>
说起,/不重写任何其他的servlet,它只替换容器里面内置默认servlet,用于所有与任何其他已注册servlet不匹配的请求;所以使用<url-pattern>/<urlpattern>
是由于jsp可以匹配到容器内置的JSPservlet所以不会交给DispatcherServlet,而html、css等等静态资源是没有匹配的内置servlet,所以交给DispatcherServlet来处理,而DispatcherServlet处理请求会找对应的HanlderMapping,静态资源没有映射所以404(这就是造成问题的关键);如果使用/*,它会匹配所有请求,所以找jsp也会报错
解决方法
<mvc:resources mapping="" location=""/>
:手动配置映射,这样请求到HandlerMapping的时候会转到ResourceHttpRequestHandler处理并返回,mapping是前端的url映射,location是本地资源路径,必须是webapp根目录下的,不能在WEB-INF下推荐<mvc:default-servlet-handler/>
:将在HandlerMapping 匹配不到的资源交由默认的servlet处理,然后由默认的servlet(在tomcat容器的web.xml配置下面),这个servlet会从根路径下面找资源
SpringMVC整合JSP
使用原生servlet
按照上文中的解决方法配置后,对于处理器找jsp时可以返回对应jsp文件的路径,该请求会直接交由jspservlet来执行而不给前端控制器
代码演示
- 配置页面的映射:
<mvc:resources mapping="/pages/**" location="/pages/"/>
- 返回页面路径:
@GetMapping("getJSP")
public String getJSP(){
return "/pages/test.jsp";
}
注意这里的类注解要用Controller,不能用RestController
使用springmvc视图解析器
在容器中配置视图解析器:<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"></bean>
,在返回视图的时候返回对应视图的逻辑名(不带路径,不带后缀),然后具体找哪一个jsp由解析器来处理
视图解析器的配置
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/pages/"/>
<property name="suffix" value=".jsp"/>
</bean>
返回视图时会根据逻辑名查找这个路径下面是否有对应的jsp文件
Springmvc整合Thymeleaf
由于Springmvc中没有包含Thymeleaf,所以需要引入Thymeleaf-spring框架,并配置模板解析器、模板引擎、视图解析器
Thymeleaf配置
<!--模板解析器-->
<bean id="templateResolver" class="org.thymeleaf.spring5.templateresolver.SpringResourceTemplateResolver">
<property name="prefix" value="/WEB-INF/templates/"/>
<property name="suffix" value=".html"/>
<property name="templateMode" value="HTML5"/>
<property name="cacheable" value="false"/>
<property name="characterEncoding" value="UTF-8"/>
</bean>
<!--模板引擎-->
<bean id="templateEngine" class="org.thymeleaf.spring5.SpringTemplateEngine">
<property name="templateResolver" ref="templateResolver"/>
</bean>
<!--模板视图解析器-->
<bean class="org.thymeleaf.spring5.view.ThymeleafViewResolver">
<property name="templateEngine" ref="templateEngine"/>
<property name="characterEncoding" value="UTF-8"/>
</bean>
如果存在多视图解析器,需要分别指定viewNames属性,不然会找到第一个配置的视图解析器
数据共享
Model
- addAttribute(k,v):在model中添加值,并指定值的名字
- addAttribute(obj):在mode中添加值,值的名字时值的类型名
model的数据默认存在request域中
Model是一个接口,传入的Model对象是一个子类型BindingAwareModelMap,在它上层还有Map等夫类型,任何一个它的父类对象都可以这里作为参数载体(使用model最便捷)
@SessionAttributes和@SessionAttribute
- 在原生的Servlet中,想要将数据存在session域中,需要使用到HttpSession对象,springmvc也支持这样的写法,但是提供了封装好的方法
- @SessionAtttibutes是一个类注解,value是一个字符串数组指定变量名,表示这些变量将存在于session域中,之后使用Model存的对应变量将会存在session域而非request域中
- @SessionAttribute是一个参数注解,表示将一个session域中的变量赋值给这个参数
@ModelAttribute
- 如果@ModelAttribute修饰方法,那么将会绑定该方法的返回值(方法的返回值会被存到model中),而且该方法在每一个请求来的时候都会被调用(只有该方法没有被映射时生效)
- 如果@ModelAttribute修饰参数,如果这个参数是一个简单类型(基本类型、字符串、日期、URI、URL等)将会传入Model域中的值而非传来的参数,如果是一个复杂类型,将会以@ModelAttribute处理,这个时候会将复杂类型对象绑定Model
应用
页面提交表单时,如果出现校验失败的情况,所有表单项提交之后都会被清楚内容,用户体验感差,通过Model就可以轻松的将用户输入的数据重现在表单上
- 方法一:
接收参数后,使用将参数存在model中,然后jsp上调用数据域中的数据 - 方法二:
对于一个对象,传参的时候它会将其域model绑定,省去很多操作
RedirectAttributes
作为参数类型,其中的属性跟在重定向的url后或替换其中的占位符