欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

SpringMVC框架--学习笔记(上)

程序员文章站 2022-06-08 19:06:10
...

1、SpringMVC入门程序:

(1)导入jar包:spring核心jar包、spring-webmvc整合Jar包

SpringMVC框架--学习笔记(上)

(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能让哪些映射器处理,就让正确的映射器处理。

SpringMVC框架--学习笔记(上)


SpringMVC框架--学习笔记(上)

//编写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方法的形参名称一致。

SpringMVC框架--学习笔记(上)


8、参数绑定:简单类型的pojo绑定:

页面中input的name和controller的pojo形参中的属性名称一致,将页面中数据绑定到pojo。

SpringMVC框架--学习笔记(上)

SpringMVC框架--学习笔记(上)


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:

SpringMVC框架--学习笔记(上)

SpringMVC框架--学习笔记(上)

SpringMVC框架--学习笔记(上)


11、参数绑定:数组绑定:

SpringMVC框架--学习笔记(上)

SpringMVC框架--学习笔记(上)


12、参数绑定:list绑定:

通常在需要批量提交数据时,将提交的数据绑定到list<pojo>中,比如,成绩录入(录入多门成绩,批量提交)。

SpringMVC框架--学习笔记(上)

SpringMVC框架--学习笔记(上)

SpringMVC框架--学习笔记(上)


13、参数绑定:Map绑定:

在包装类中定义Map对象,并添加get/set方法,action使用包装对象接收。

包装类中的定义Map对象如下:

SpringMVC框架--学习笔记(上)

SpringMVC框架--学习笔记(上)


14、validation校检:

项目中,通常使用较多的是前端的校检,比如页面中的js校检。对于安全要求较高点建议在服务端进行校验。

(1)服务端检验:

        控制层controller,校验页面请求的参数的合法性。在服务端控制层controller校验,不区分客户端类型(浏览器、手机客户端、远程调用)。

        业务层service(使用较多):主要检验关键业务参数,仅限于service接口中使用的参数。

        持久层dao:一般不进行校验。

(2)springmvc校验:

springmvc使用hibenate的校验框架(和hibernate没有任何关系)

校验思路:页面提交请求参数,请求到controller方法中,使用validation进行校验。如果检验出错,将错误信息展示到页面。

步骤:

①导入jar包:

hibernate-validator.5.x.xvalidation-api.1.0.0版本冲突,使用hibernate-validator.4.x.x

SpringMVC框架--学习笔记(上)

②配置校验器:(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)在页面中显示错误信息:

SpringMVC框架--学习笔记(上)


15、validation分组检验:

每个controller方法需要不同的校验。

解决方法:定义多个校验分组(其实是一个java接口),分组中定义有哪些规则,每个controller方法使用不同的校验分组。

(1)创建分组接口类:

public interface ValidGroup1 {
	//接口不需要定义任何方法,仅是对不同的校验规则进行分组
	//此分组只对商品名称长度进行校验

}

(2)定义分组:(使用groups)

SpringMVC框架--学习笔记(上)

(3)在controller指定使用哪个分组:

SpringMVC框架--学习笔记(上)


16、数据回显:

(1)简单类型的数据回显:使用model的方法。

model.addAttribute("id",id);

(2)pojo类型的数据回显:

第一种:默认方式:

pojo数据传入controller方法后,springmvc自动将pojo数据放到request域,key等于pojo类型(首字母小写)

第二种:使用@ModelAttribute:指定pojo回显到页面在request中的key。

SpringMVC框架--学习笔记(上)

aaa@qq.com还可以将方法的返回值传到页面:

SpringMVC框架--学习笔记(上)

页面上可以得到itemTypes数据:

SpringMVC框架--学习笔记(上)

第三种:使用model方式:

//可以直接使用model将提交pojo回显到页面
model.addAttribute("items",itemsCustom);


接下篇:SpringMVC框架--学习笔记(下):https://blog.csdn.net/a745233700/article/details/81045960