SpringMvc初级
三层架构
-
表现层:负责数据展示 WEB
-
业务层:负责业务处理 BLL
-
数据层:负责数据操作 DAL
概念*
-
Spring MVC 是Spring提供的一个实现了Web MVC设计模式的轻量级Web框架。
-
MVC(Model View Controller),一种用于设计创建Web应用程序表现层的模式
- Model(模型):数据模型,用于封装数据
- View(视图):页面视图,用于展示数据
- Controller(Handle 处理器):处理用户交互的调度器,用于根据用户需求处理程序逻辑
工作流程分析
- 服务器启动
- 加载web.xml中DispatcherServlet
- 读取spring-mvc.xml中的配置,加载所有com.xinzhi包中所有标记为bean的类
- 读取bean中方法上方标注@RequestMapping的内容
- 处理请求
- DispatcherServlet配置拦截所有请求 /
- 使用请求路径与所有加载的@RequestMapping的内容进行比对
- 执行对应的方法
- 根据方法的返回值在webapp目录中查找对应的页面并展示
-
web三大组件有 处理器映射,处理器适配器, 视图解析器
1 dispatcherServlet 前置控制器,负责接收并处理所有的web请求,根据handlerMapping(处理器映射)找到具体的Controller(处理器),由controller完成具体的处理逻辑。
2 HandlerMapping(处理器映射器):负责处理web请求和具体的Controller之间的映射关系匹配。
3HandlerAdapter(处理器适配器) 通过 HandlerAdapter 对处理器进行执行,这是适配器模式的应用,通过扩展适配器可以对更多类型的处理器进行执行。 主要处理方法参数、相关注解、数据绑定、消息转换、返回值、调用视图解析器等等。
4.Controller(处理器):DispatherServlet的次级控制器,web请求的具体处理者。DispatherServlet获得handlerMapping的返回结果后,调用controller的处理方法处理当前的业务请求,处理完成后返回ModelAndView对象。
5 ViewResolver( 视图解析器):用来处理视图名与具体的view实例之间的映射对应关系。根据ModelAndView中的视图名查找相应的View实现类,然后将查找的结果返回给DispatcherServlet,DispatcherServlet最终会将ModelAndView中的模型数据交给返回的View处理最终的视图渲染工作。
Springmvc架构原理解析
第一步:发起请求到前端控制器(DispatcherServlet)
第二步:前端控制器请求HandlerMapping查找 Handler,可以根据xml配置、注解进行查找
第三步:处理器映射器HandlerMapping向前端控制器返回Handler
第四步:前端控制器调用处理器适配器去执行Handler
第五步:处理器适配器去执行Handler
第六步:Handler执行完成给适配器返回ModelAndView
第七步:处理器适配器向前端控制器返回ModelAndView
ModelAndView是springmvc框架的一个底层对象,包括 Model和view
第八步:前端控制器请求视图解析器去进行视图解析
根据逻辑视图名解析成真正的视图(jsp)
第九步:视图解析器向前端控制器返回View
第十步:前端控制器进行视图渲染
视图渲染将模型数据(在ModelAndView对象中)填充到request域
第十一步:前端控制器向用户响应结果
请求参数的绑定
1 默认类型:
直接放在参数上就可以使用的数据,HttpServletRequest
2 简单类型:
直接将简单类型的数据放在方法里,如果前端参数和后端参数名字一样,自动匹配;
名字不一样:@RequsetParam(“前端的值”) 就可以将前传的值和后端参数映射
3 对象 :
前端的参数要和对象的属性名称必须一致,会自动封装。
4 对象嵌套:
参数和对象的属性名称一致,前端参数对象子属性必须(子对象.属性)
5 自定义数据的绑定
5.1 编写转换 器类,作用是将前端的数据类型转换成后端的数据类型,继承converter
5.2 配置文件中,添加转化器驱动
6 数组
前端数组中是简单类型的数据,那么前端数组中的name要和后端数组名称一致
7 集合
后端接受的对象是含有List<对象>属性的,那么前端的name值格式要和后端list属性名称一致,而且用索引的格式 list[0].属性(list集合里对象的属性名称)
//默认类型
@RequestMapping("/say02")
public ModelAndView say02(HttpServletRequest request, ModelAndView modelAndView){
String name = request.getParameter("name");
System.out.println(name);
modelAndView.setViewName("b.jsp");
modelAndView.addObject("name",name);
return modelAndView;
}
//简单类型
@RequestMapping("/say03")
public String say03(String name,int age){
System.out.println(name);
System.out.println(age);
return "b.jsp";
}
//对象类型
@RequestMapping("/say04")
public String say04(User user){
System.out.println(user);
return "b.jsp";
}
//数组
@RequestMapping("/say05")
public String say05(String[] hobby){
System.out.println(Arrays.toString(hobby));
return "b.jsp";
}
//list类型
@RequestMapping("/say06")
public String say06(@RequestParam("hobby") List<String> hobby){
System.out.println(hobby);
return "b.jsp";
}
-
自定义数据绑定
- 定义转换器
public class MyDateConverter implements Converter<String, Date> { @Override public Date convert(String s) { DateFormat df = new SimpleDateFormat("yyyy-MM-dd"); Date date = null; try { date = df.parse(s); } catch (ParseException e) { e.printStackTrace(); } return date; } }
- 注解驱动,使转换器起作用
<!--1.将自定义Converter注册为Bean,受SpringMVC管理--> <bean id="myDateConverter" class="com.xinzhi.converter.MyDateConverter"/> <!--2.设定自定义Converter服务bean--> <bean id="conversionService" class="org.springframework.context.support.ConversionServiceFactoryBean"> <!--3.注入所有的自定义Converter,该设定使用的是同类型覆盖的思想--> <property name="converters"> <!--4.set保障同类型转换器仅保留一个,去重规则以Converter<S,T>的泛型为准--> <set> <!--5.具体的类型转换器--> <ref bean="myDateConverter"/> </set> </property> </bean> <!--开启注解驱动,加载自定义格式化转换器对应的类型转换服务--> <mvc:annotation-driven conversion-service="conversionService"/>
//自定义date
@RequestMapping("/say07")
public String say07(Date date){
System.out.println(date);
return "b.jsp";
}