SpringMVC框架--学习笔记(上)
1、SpringMVC入门程序:
(1)导入jar包:spring核心jar包、spring-webmvc整合Jar包
(2)配置前端控制器:web.xml文件中
<?xml version="1.0" encoding="UTF-8"?>
<web-app id="WebApp_9" version="2.4" xmlns="http://java.sun.com/xml/ns/j2ee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">
<display-name>Springmvc_01</display-name>
<!-- 配置前端控制器 -->
<servlet>
<servlet-name>springmvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<!-- contextConfigLocation:配置springmvc加载的配置文件(处理处映射器、处理器适配器等等) -->
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:springmvc.xml</param-value>
</init-param>
</servlet>
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<!--
第一种:*.action,访问以.action结尾由DispatcherServlet进行解析
第二种:/,所有访问的地址都由DispatcherServlet进行解析,对于静态文件的解析需要配置
不让DispatcherServlet进行解析,使用此种方法可以实现RESTful风格的url
第三种:/*,这种配置不对,使用这种配置,最终要转发到一个jsp页面,仍然会由
DispatcherServlet解析jsp,不能根据jsp页面找到Handle,会报错-->
<url-pattern>*.action</url-pattern>
</servlet-mapping>
<welcome-file-list>
<welcome-file>index.html</welcome-file>
<welcome-file>index.htm</welcome-file>
<welcome-file>index.jsp</welcome-file>
<welcome-file>default.html</welcome-file>
<welcome-file>default.htm</welcome-file>
<welcome-file>default.jsp</welcome-file>
</welcome-file-list>
</web-app>
(3)springmvc.xml约束:
(4)配置处理器映射器:
(5)配置处理器适配器:
(6)配置视图解析器:
(7)配置Handle:
(8)编写Handle:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd">
<!-- 非注解:配置Handler
作用:将编写的Handler在spring容器加载
<bean id="ItemsController1" name="/queryItems.action" class="com.zwp.controller.ItemsController1"></bean>
<bean id="ItemsController2" class="com.zwp.controller.ItemsController2"></bean>
非注解的处理器映射器
第一种:将bean的name作为url进行查找,需要配置Handler是指定beanname(就是url)
<bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping"></bean>
第二种:简单url映射
<bean class="org.springframework.web.servlet.handler.SimpleUrlHandlerMapping">
<property name="mappings">
<props>
对ItemsController1进行url映射,
url是key的值,ItemsController1是Handler的id值
<prop key="/queryItems1.action">ItemsController1</prop>
<prop key="/queryItems11.action">ItemsController1</prop>
<prop key="/queryItems2.action">ItemsController2</prop>
</props>
</property>
</bean>
非注解的处理器适配器
第一种:要求handler实现Controller接口
<bean class="org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter"></bean>
第二种: 要求handler实现HttpRequestHandler接口
<bean class="org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter"></bean> -->
<!-- 注解方式:配置Handler -->
<!-- 单个加载方式: -->
<!-- <bean class="com.zwp.controller.ItemsController3"></bean> -->
<!-- 批量加载:组件扫描方式: -->
<context:component-scan base-package="com.zwp.controller"></context:component-scan>
<!-- 注解的处理器映射器(配对使用)-->
<!-- <bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping"></bean> -->
<!-- 注解的处理器适配器(配对使用) -->
<!-- <bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"></bean> -->
<!-- 实际使用 mvc:annotation-driven代替上面注解适配器和映射器
并且默认加载了很多参数绑定方法,比如json转化解析器被默认加载了-->
<mvc:annotation-driven></mvc:annotation-driven>
<!-- 视图解析器
解析jsp页面,默认使用jstl标签,classpath下得有jstl的包 -->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/jsp/"></property>
<property name="suffix" value=".jsp"></property>
</bean>
</beans>
//编写Hander:实现controller接口:
public class ItemsController1 implements Controller{
@Override
public ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response) throws Exception {
//调用service查询数据库,查询商品列表,这里使用静态数据
List<Items> itemsList=new ArrayList<Items>();
Items items1=new Items();
items1.setName("键盘");
items1.setPrice(80.0f);
items1.setDetail("机械键盘");
Items items2=new Items();
items2.setName("苹果手机");
items2.setPrice(4000.0f);
items2.setDetail("iphone6");
itemsList.add(items1);
itemsList.add(items2);
//返回modelAndView
ModelAndView modelAndView=new ModelAndView();
//相当于request的setAttribute,在jsp页面通过itemsList取数据
modelAndView.addObject("itemsList", itemsList);
//指定视图
modelAndView.setViewName("/WEB-INF/jsp/Items/itemsList.jsp");
return modelAndView;
}
}
至此,一个简单的入门springmvc程序就编写完成,只需要在网页访问响应的地址就可以了,我的访问地址是:
http://localhost:8080/Springmvc_01/queryItems.action
2、非注解的处理器映射器和适配器:
多个映射器可以共存,前端控制器判断url能让哪些映射器处理,就让正确的映射器处理。
//编写Hander:使用第二种处理器适配器,需要实现HttpRequestHandler接口:
public class ItemsController2 implements HttpRequestHandler{
@Override
public void handleRequest(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
List<Items> itemsList=new ArrayList<Items>();
Items items1=new Items();
items1.setName("键盘");
items1.setPrice(80.0f);
items1.setDetail("机械键盘");
Items items2=new Items();
items2.setName("苹果手机");
items2.setPrice(4000.0f);
items2.setDetail("iphone6");
itemsList.add(items1);
itemsList.add(items2);
//设置模型数据
request.setAttribute("itemsList", itemsList);
//设置视图:
request.getRequestDispatcher("/WEB-INF/jsp/Items/itemsList.jsp").forward(request, response);
//此方法可以通过修改response,设置响应的数据格式,比如json数据
// response.setCharacterEncoding("utf-8");
// response.setContentType("application/json;charset=utf-8");
// response.getWriter().write("json串");
}
}
3、注解的处理器映射器和适配器(配对使用):
<!-- 注解的处理器映射器(配对使用)-->
<!-- <bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping"></bean> -->
<!-- 注解的处理器适配器(配对使用) -->
<!-- <bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"></bean> -->
<!-- 实际使用 mvc:annotation-driven代替上面注解适配器和映射器
并且默认加载了很多参数绑定方法,比如json转化解析器被默认加载了-->
<mvc:annotation-driven></mvc:annotation-driven>
(1)开发Handler:
//@Controller:表明这是一个控制器,不需要实现任何接口
@Controller
public class ItemsController3{
//使用@RequestMapping注解,实现方法名和url进行映射,一个方法对应一个url
//建议方法名和url一样
@RequestMapping("queryItems")
public ModelAndView queryItems() throws Exception {
//调用service查询数据库,查询商品列表,这里使用静态数据
List<Items> itemsList=new ArrayList<Items>();
Items items1=new Items();
items1.setName("键盘");
items1.setPrice(80.0f);
items1.setDetail("机械键盘123");
Items items2=new Items();
items2.setName("苹果手机");
items2.setPrice(4000.0f);
items2.setDetail("iphone6s");
itemsList.add(items1);
itemsList.add(items2);
//返回modelAndView
ModelAndView modelAndView=new ModelAndView();
//相当于request的setAttribute,在jsp页面通过itemsList取数据
modelAndView.addObject("itemsList", itemsList);
//指定视图
modelAndView.setViewName("Items/itemsList");
return modelAndView;
}
}
(2)在springmvc中加载Handler:
<!-- 注解方式:配置Handler -->
<!-- 单个加载方式: -->
<!-- <bean class="com.zwp.controller.ItemsController3"></bean> -->
<!-- 批量加载:组件扫描方式: -->
<context:component-scan base-package="com.zwp.controller"></context:component-scan>
4、视图解析器:前缀和后缀:
<!-- 视图解析器
解析jsp页面,默认使用jstl标签,classpath下得有jstl的包 -->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/jsp/"></property>
<property name="suffix" value=".jsp"></property>
</bean>
5、RequestMappering的作用:
(1)用在类上面:窄化url;
(2)用上方法上面:映射url路径;
(3)用在方法上面:通过method属性,限制http请求。
6、Controller三种返回值:ModelAndView、String、void:
(1)返回ModelAndView:
需要在方法结束时,定义ModelAndView,将model和view分别进行设置。
//controller第一种返回值:ModelAndView
//商品查询列表
@RequestMapping("/queryItems")
public ModelAndView queryItems(HttpServletRequest request,ItemsQueryVo itemsQueryVo) throws Exception {
//调用service查询数据库,查询商品列表
List<ItemsCustom> itemsList=itemsService.selectItemsList(itemsQueryVo);
//返回modelAndView
ModelAndView modelAndView=new ModelAndView();
//相当于request的setAttribute,在jsp页面通过itemsList取数据
modelAndView.addObject("itemsList", itemsList);
//指定视图
modelAndView.setViewName("Items/itemsList");
return modelAndView;
}
(2)返回String:
如果controller方法返回String,表示返回逻辑视图名。真正视图(jsp路径)=前缀+逻辑视图名+后缀。
//Controller第二种返回值:String
//@RequestParam里边指定指定requset传入参数和形参进行绑定
//通过required属性指定参数是否必须要传入;通过defaultValue可以设置默认值,如果id参数没有传入,将默认值和形参绑定
//商品信息修改页面显示:
@RequestMapping(value="/editItems",method={RequestMethod.POST,RequestMethod.GET})
public String editItems(Model model,@RequestParam(value="id",required=true) Integer items_id) throws Exception
{
//调用service查询数据库,根据id查询商品信息
ItemsCustom itemsCustom=itemsService.findItemsById(items_id);
if(itemsCustom==null)
{
throw new CustomException("修改的商品信息不存在");
}
//通过形参中的model将model传到页面上;
model.addAttribute("itemsCustom", itemsCustom);
//返回逻辑视图名
return "Items/editItems";
}
--redirect从定向:
特点:浏览器地址栏中的url会变化,修改提交的request数据无法传到重定向地址中。
--forward页面转发:
特点:通过forward进行页面转发,浏览器地址栏url不变,request可以共享。
//重定向:
return "redirect:queryItems.action";
//页面转发:
return "forward:queryItems.action";
(3)void:
在controller方法形参上可以定义request和response,使用request和response指定响应结果:
①使用request转向页面,如下:
request.getRequestDispatcher("页面路径").forward(request,response);
②通过response页面重定向:
response.sendRedirect("url");
③也可以通过response指定响应结果,例如响应json数据如下:
response.setCharacterEncoding("utf-8");
response.setContentType("application/json;charset=utf-8");
response.getWriter().write("json串");
7、参数绑定:
(1)参数绑定过程:从客户端请求key/value数据,经过参数绑定,将key/value数据绑定到controller方法的形参上。
springmvc中,接收页面提交的数据是通过方法形参来接收,而不是像struts2在controller类定义成员变量接收。
(2)默认支持的类型:
直接在controller方法形参上定义下边类型的对象,就可以使用这些对象。在参数绑定过程中,如果遇到下边类型直接进行绑定:
①HttpServletRequest:通过request对象获取请求信息;
②HttpServletResponse:通过response处理响应信息;
③HttpSession:通过session对象得到session中存放的对象;
④Model/ModelMap:model是一个接口,modelmap是一个接口实现;作用:将model数据填充到request域。
⑤还支持一些简单类型的参数绑定:通过@PequestParam对简单类型的参数进行绑定。
如果不使用@PequestParam,要求request传入的参数名称和controller方法的形参名称一致,方可绑定成功。
如果使用@PequestParam,不限制request传入的参数名称和controller方法的形参名称一致。
8、参数绑定:简单类型的pojo绑定:
页面中input的name和controller的pojo形参中的属性名称一致,将页面中数据绑定到pojo。
9、自定义参数绑定:
--自定义参数绑定实现日期类型绑定:
对于controller形参中的pojo对象,如果属性中有日期类型,需要自定义参数绑定。
将请求日期数据串转成日期类型,要转换的日期类型和pojo中日期属性类型保持一致。
(1)springmvc.xml配置文件:
<!--conversion-service="conversionService:注入自定义参数绑定" -->
<mvc:annotation-driven conversion-service="conversionService"></mvc:annotation-driven>
<!-- 自定义参数绑定 -->
<bean id="conversionService" class="org.springframework.format.support.FormattingConversionServiceFactoryBean">
<!-- 转化器 -->
<property name="converters">
<list>
<!-- 日期类型装换器 -->
<bean class="com.zwp.ssm.controller.converter.CustomDateConverter"></bean>
</list>
</property>
</bean>
(2)自定义的绑定器:
//自定义参数绑定:
//实现啊Converter接口,<转化前格式,转化后格式>,
//转化后的格式要和接收的pojo中的属性类属性一致
public class CustomDateConverter implements Converter<String,Date>{
@Override
public Date convert(String source) {
//自定义日期格式绑定器:
SimpleDateFormat simpleDateFormat=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
try{
//转换成功直接返回
return simpleDateFormat.parse(source);
}catch(Exception e){
e.printStackTrace();
}
//如果参数绑定失败,返回null
return null;
}
}
10、参数绑定:包装类型的pojo:
11、参数绑定:数组绑定:
12、参数绑定:list绑定:
通常在需要批量提交数据时,将提交的数据绑定到list<pojo>中,比如,成绩录入(录入多门成绩,批量提交)。
13、参数绑定:Map绑定:
在包装类中定义Map对象,并添加get/set方法,action使用包装对象接收。
包装类中的定义Map对象如下:
14、validation校检:
项目中,通常使用较多的是前端的校检,比如页面中的js校检。对于安全要求较高点建议在服务端进行校验。
(1)服务端检验:
控制层controller,校验页面请求的参数的合法性。在服务端控制层controller校验,不区分客户端类型(浏览器、手机客户端、远程调用)。
业务层service(使用较多):主要检验关键业务参数,仅限于service接口中使用的参数。
持久层dao:一般不进行校验。
(2)springmvc校验:
springmvc使用hibenate的校验框架(和hibernate没有任何关系)
校验思路:页面提交请求参数,请求到controller方法中,使用validation进行校验。如果检验出错,将错误信息展示到页面。
步骤:
①导入jar包:
(hibernate-validator.5.x.x和validation-api.1.0.0版本冲突,使用hibernate-validator.4.x.x)
②配置校验器:(springmvc.xml文件中)
<!-- 校验器: -->
<bean id="validator" class="org.springframework.validation.beanvalidation.LocalValidatorFactoryBean">
<!-- hibernate校验器 -->
<property name="providerClass" value="org.hibernate.validator.HibernateValidator"></property>
<!-- 指定校验使用的资源文件,如果不指定则默认使用classpath下的ValidationMessage.properties -->
<property name="validationMessageSource" ref="messageSource"></property>
</bean>
<!-- 校验错误信息配置文件 -->
<bean id="messageSource" class="org.springframework.context.support.ReloadableResourceBundleMessageSource">
<!-- 资源文件名 -->
<property name="basenames">
<list>
<value>classpath:CustomValidationMessage</value>
</list>
</property>
<!-- 资源文件编码格式 -->
<property name="fileEncodings" value="utf-8"></property>
<!-- 对资源文件内容缓存时间,单位秒 -->
<property name="cacheSeconds" value="120"></property>
</bean>
(3)检验器注入到处理器适配器中:
<!-- validator="validator":注入校验器 -->
<mvc:annotation-driven validator="validator"></mvc:annotation-driven>
(4)错误信息文件:CustomValidationMessage.properties#添加校验错误提交信息
items.name.length.error=请输入1到30个字符的商品名称
items.creatime.isNull=生产日期不能为空
(5)在pojo中添加校验规则:
//在pojo中添加校验规则
public class Items {
private Integer id;
//校验名称在1-30个字符之间,message是校验出错的显示信息
//groups:定义校检器分组,可以定义多个分组
@Size(min=1,max=30,message="{items.name.length.error}",groups={ValidGroup1.class})
private String name;
private Float price;
private String pic;
//校验生产日期不能为空
@NotNull(message="{items.creatime.isNull}")
private Date creatime;
private String detail;
}
(6)在controller中捕获错误信息:
//商品信息修改提交
//在需要校检的poji前面加上@Validated,在需要校检的pojo后面加上BindingResult bindingResult接收校检出错的信息
//Validated和BindingResult必须配对出现,且顺序不能改变
@RequestMapping(value="/editItemsSubmit",method={RequestMethod.POST,RequestMethod.GET})
public String editItemsSubmit(Model model,HttpServletRequest request,Integer id,
@ModelAttribute("itemsCustom") @Validated(value={ValidGroup1.class}) ItemsCustom itemsCustom,BindingResult bindingResult,
MultipartFile items_pic) throws Exception
{
if(bindingResult.hasErrors())
{
List<ObjectError> allErrors=bindingResult.getAllErrors();
//自定义一个list接受自己编码后的提示字符串,在把自己定义的list传到界面,
//这样就解决了把乱码传到界面的问题了
List<String> listErrors=new ArrayList<>();
for(ObjectError objectError:allErrors)
{
//System.out.println(objectError.getDefaultMessage());
//把返回错误的提示再次编码
String strError=new String(objectError.getDefaultMessage().getBytes("ISO-8859-1"),"UTF-8");
listErrors.add(strError);//把编码好的错误提示信息加自己定义好list集合里面去
}
model.addAttribute("allErrors", listErrors);
return "Items/editItems";
}
return "forward:queryItems.action";
}
(7)在页面中显示错误信息:
15、validation分组检验:
每个controller方法需要不同的校验。
解决方法:定义多个校验分组(其实是一个java接口),分组中定义有哪些规则,每个controller方法使用不同的校验分组。
(1)创建分组接口类:
public interface ValidGroup1 {
//接口不需要定义任何方法,仅是对不同的校验规则进行分组
//此分组只对商品名称长度进行校验
}
(2)定义分组:(使用groups)
(3)在controller指定使用哪个分组:
16、数据回显:
(1)简单类型的数据回显:使用model的方法。
model.addAttribute("id",id);
(2)pojo类型的数据回显:
第一种:默认方式:
pojo数据传入controller方法后,springmvc自动将pojo数据放到request域,key等于pojo类型(首字母小写)
第二种:使用@ModelAttribute:指定pojo回显到页面在request中的key。
aaa@qq.com还可以将方法的返回值传到页面:
页面上可以得到itemTypes数据:
第三种:使用model方式:
//可以直接使用model将提交pojo回显到页面
model.addAttribute("items",itemsCustom);
接下篇:SpringMVC框架--学习笔记(下):https://blog.csdn.net/a745233700/article/details/81045960
上一篇: 喝酒多了伤肝吃什么好
下一篇: 如何做肉丸,你知道吗