SpringMVC体系结构and处理请求控制器

2024-08-06 13:13:32 浏览数 (2)

SpringMVC

MVC设计模式:

首先先熟悉回忆一下MVC设计模式, 了解

MVC分层设计模式:

它是软件架构模式的一种, 强制的将软件系统的: 输入 处理 输出 把软件系统分为三个部分:模型(Model) 视图(View) 控制器(Controller)

视图(View): 负责: 数据展示 用户交互 数据验证 界面设计 …等功能; 组件: JSP 或 HTML文件… 控制器(Controller): 负责:接收并转发请求,对请求进行处理,做出对于的响应操作; 组件:Servlet… 模型(Model):是应用程序的主体部分 负责:数据业务逻辑操作处理, 实现数据的存取操作… JavaBean(Java类) 组件:业务逻辑(Service) 与数据库交互(Dao) 贯穿各层的数据模型,实体类(POJO/以前我都是entity)


JSP Model1

只有视图 和 模型…

当业务流程为简单的时候,可以把控制器的功能交给视图来实现, 这种模式被称为 JSP Model1 总结: Model1 在一定基础上,实现了MVC :JSP( 控制层和视图层 ) JavaBean为模型层; 其中JSP 身兼数职, 又要负责数据展示, 还要注意 业务流程控制, 结构较为混乱… 而且 也不是程序适合的 松耦合架构模式 当业务流程复杂时候不推荐使用

JSP Model2

这种模式就是 JSP Servlet JavaBean (哈哈哈,以前学过现在在学框架,有点忘记了有对这方面的笔记,在oneNote上… 还做了一个小型电商项目)

相比 Model1 , Model2是将控制层(Servlet)单独划分出来负责业务流程的控制, 接收请求 创建所需的JavaBean实例; 并将处理后的数据,响应给视图层(JSP) 总结: 相比 Model1 , Model2结构更清晰 JSP不在一个人抗下所有 是一个相对 松耦合 的架构模式; 所以除非项目非常简单使用 Model1, 一般都使用 Model2

MVC处理过程

MVC 优点 多视图共享一个模型,大大提高代码的可重用性 MVC三个模块相互独立,松耦合架构 控制器提高了应用程序的灵活性和可配置性 有利于软件工程化管理 完美的系统架构 = 松耦合 高重用性 高扩展性 MVC 缺点 原理复杂 增加了系统结构和实现的复杂性 视图对模型数据的低效率访问 (中间还要经过一个控制器~必定会影响…) so:它并不适合, 小型项目,花费大量时间将 MVC应用到并不是很大的 应用程序中通常 "得不偿失"

智勇建议你可以:细品细品

SpringMVC 介绍及环境搭建:

ok, 了解了MVC设计模式之后就可以更容易的, 接收SpringMVC 框架了 SpringMCV 就是 Spring框架提供一个用于 Web应用开发中的一个框架; SpringMVC框架介绍: 在MVC设计模式中, SpringMVC 就是作为控制器( Controller ) 来建立模型与视图的数据交互; 结构最清晰的MVC Model2实现 SpringMVC 框架采用松耦合 可拔插的组件结构, 相比其它 MVC框架 ,具有高度可扩展性;

SpringMVC环境搭建:

在MyElicpse 中新建Web ProJect项目后配置 SpringMVC框架; Spring MVC框架搭建步骤: 下载jar文件并导入工程 :(Myelicpse工具有自带的类库~ ) spring-web-3.2.13.RELEASE.jar Web应用开发的使用 Spring框架所需的 核心类; spring-webmvc-3.2.13.RELEASE.jar 框架相关的类,包含框架的 Servlets WebMVC 以及对控制器 和 视图的支持; 配置文件在web.xml中配置< Servlet> 元素; 创建Spring MVC的配置文件(也是Spring 核心配置文件) 创建Controller-处理请求的控制器BeanNameUrlHandlerMapping (相当于以前写的 Servlet ;) 创建View-JSP 部署运行

正片开始!

项目连接

配置文件: 在web.xml中配置< Servlet> 元素; web.xml

代码语言:javascript复制
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" version="3.0">

	<!-- 配置SpringMVC核心控制器:DispatcherServlet
			DispatcherServlet:是SpringMVC的核心,负责接收请求 和 响应操作; 
	 -->
	<servlet>
		<!-- 
			servlet-name: 声明一个名为:mvc的Servlet 
			servlet-class: 类型是DispatcherServlet,注意别导错包咯~
		 -->
		<servlet-name>mvc</servlet-name>
		<servlet-class>org.springframework.web.servlet.DispatcherServlet </servlet-class>
		<!-- 初始化参数:
			通过contextConfigLocation 属性来指定SpringMVC配置文件的位置;
		 -->
		<init-param>
			<param-name>contextConfigLocation</param-name>
			<param-value>classpath:applicationContext.xml</param-value>
		</init-param>
		<!-- 值 1 : 配置标记服务器启动时候加载DispatcherServlet  -->
		<load-on-startup>1</load-on-startup>
	</servlet>
	
	<!-- 通过servlet-mapping: 指定对应的Servlet -->
	<servlet-mapping>
		<servlet-name>mvc</servlet-name> <!-- 指定对于的Servletname -->
		<url-pattern>/</url-pattern> 	  <!-- / 匹配所有的请求(不包含.jsp)  /* 包含.jsp  即所有的请求都会经过 url-pattern -->
	</servlet-mapping>
	

</web-app>

/ 和 / * 的区别:

< url-pattern > / </ url-pattern > 不会匹配到.jsp, 只针对我们编写的请求;即:.jsp 不会进入spring的 DispatcherServlet类 。 < url-pattern > /* </ url-pattern > 会匹配 *.jsp . 因此会出现返回 jsp视图 时再次进入spring的DispatcherServlet 类,导致找不到对应的controller所以报404错。 映射路径为 / 【不要用/ *,会404】

第二步可以创建 Controller(控制器) 也可以是 SpringMVC 的核心配置文件; 我还是喜欢 Controller 其实就相当于以前的 Servlet ; 创建Controller 和Servlet 一样该控制器本质其实也是一个 JavaBean(普通的Java类) Servlet 是继承了HttpServlet 抽象类; Controller 则是,需要继承:org.springframework.web.servlet.mvc.AbstractController 类; 并实现对应的 AbstractController 方法(); HelloController.java

代码语言:javascript复制
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.mvc.AbstractController;  //导包

public class HelloController extends AbstractController {  		//继承了 AbstractController 类
	@Override													//实现handleRequestInternal方法(request,response); 返回值:ModelAndView
	protected ModelAndView handleRequestInternal(HttpServletRequest request, 	
			HttpServletResponse response) throws Exception {
		//打印输出显示
		System.out.println("进入Hello:Controller");
		//根据url name获取作用域值;
		String name = request.getParameter("name");
		name = "Hello:" name;
		//创建一个 ModelAndView 对象;
		ModelAndView mav = new ModelAndView();
		mav.addObject("msg",name);		
		mav.setViewName("index");	
 		return mav;
 		
// 		这里该控制器只处理一种控制;如果有多个还需要像之前一个 url中存在一个 opt 之类的name,来确认要做的操作;
// 		String op = request.getParameter("op");
// 		if("增".equals(op)){
// 			....省略操作
// 		}else if("删".equals(op)){
// 			....省略操作
// 		}else ....
	}
}
/*
 * ModelAndView(模块 和 视图):
 * 		正如其名,它代表 SpringMVC 中呈现视图界面所使用的 Model(模型书记) 和 View(视图) 
 * 		addObject(..);		设置需要返回的值,类似于  request.setAttribute(x,x); 键值方式存在数据在 Model中;
 * 		setViewName(..);	通过方法跳转到指定的页面名 经过 视图解析器,加上文件: 前缀/后缀 |最终响应浏览器;
 * 		addAllObjects(Map<String,?> map); 方法看出模型数据也是一个Map 对象;
 */

ok 控制器写完了就是SpringMVC的配置文件了; applicationContext.xml

代码语言:javascript复制
<?xml version="1.0" encoding="UTF-8"?>
<beans
	xmlns="http://www.springframework.org/schema/beans"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xmlns:p="http://www.springframework.org/schema/p"
	xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.1.xsd">

	<!-- 配置处理器映射
		name	:页面根据name名,找到对应的 控制器; 
		class	:指定对应的控制器;
	作用:
		将指定的URL 请求指定给一个Controller 处理;
		Spring		提供了多种处理器映射(不一一举例了)...根据需求选择合适的处理器映射;
		Spring		默认使用BeanNameUrlHandlerMapping : Spring容器根据URL名查找,同名的Bean.. 
		所以web.xml <url-pattern>/</url-pattern> 将根目录截取之后的 文件名;这里就是  /Hollo.html 了,就通过文件名,找到对应的 控制器;
	-->
	
	<bean name="/Hollo.html" class="com.wsm.controller.HelloController" />



	<!-- 配置视图解析器(ViewResolver): 
		处理请求的最后一件事情就是 "渲染输出" 控制器做出响应最后会经过这里进行渲染输出;
		DispatcherServlet(前端控制器)
			会查到一个视图解析器,将控制器返回的逻辑视图名称,渲染为一个指定的 实际视图文件上;
		Spring同样提供了多种...这里使用:InternalResourceViewResolver
		总结:
			请求处理方法执行完之后,最终会返回一个 ModelAndView 对象,对于那些返回String 等类型的处理方法;
			SpringMVC 会在内部通过ViewResolver将它们装配成一个 ModelAndView对象,它包含 "逻辑视图" "数据模型" 
	 	通常使用 InternalResourceViewResolver 作为一个视图解析器,通常用于存储 JSP 和 JSTL 等视图;		
	 -->
	 
	<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" >
		<!-- <property name="prefix" value="前缀"></property> -->
		<property name="suffix" value=".jsp"></property>
	</bean>	
	
	<!-- 
		suffix		后缀,为视图ViewName 添加后缀;
		prefix		前缀,默认文件为根目录  localhost:8080/项目名/  前缀就是 localhost:8080/项目名/前缀/文件名 后缀 
					方式返回浏览器;
		这里建议和前面控制器搭配学习...
	 -->
</beans>

最后奉上 index.jsp 可以运行了

代码语言:javascript复制
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
<% 
String path = request.getContextPath();
String basePath = request.getScheme() "://" request.getServerName() ":" request.getServerPort() path "/";
%>

<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
  <head>
    <base href="<%=basePath%>"> 
    
    <title>My JSP 'index.jsp' starting page</title>
	<meta http-equiv="pragma" content="no-cache">
	<meta http-equiv="cache-control" content="no-cache">
	<meta http-equiv="expires" content="0">    
	<meta http-equiv="keywords" content="keyword1,keyword2,keyword3">
	<meta http-equiv="description" content="This is my page">
	<!--
	<link rel="stylesheet" type="text/css" href="styles.css">
	-->
  </head>
  
  <body>
     <h2>${msg }</h2>
	<form action="Hollo.html" method="post">
		<input type="text" name="name">
		<input type="submit" value="提交">
	</form>
  </body>
</html>

验证, 提交后程序经过了 控制器处理后,再次呈现给用户看…

注解操作 参数传递;

上述示例通过 BeanNameUrlHandlerMapping 访问完成了请求与 Contorller 之间的映射关系; 那如果存在很多映射则就要写很多的 … Spring容器提供了注解~ 修改Demo… applicationContext.xml

代码语言:javascript复制
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xmlns:p="http://www.springframework.org/schema/p"
	xmlns:context="http://www.springframework.org/schema/context"
	xmlns:mvc="http://www.springframework.org/schema/mvc"
	xsi:schemaLocation="http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.1.xsd
		http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.1.xsd
		http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.1.xsd">

	<!-- 使用注解完成,一次需要导入对应的命名空间:context mvc 别忘了 -->
	<context:component-scan base-package="com.wsm.controller" />  <!-- 扫描包下注解 -->
	<!--
   		支持mvc注解驱动
       		在spring中一般采用@RequestMapping注解来完成映射关系
       		要想使@RequestMapping注解生效
      		 必须向上下文中注册DefaultAnnotationHandlerMapping
       		和一个AnnotationMethodHandlerAdapter实例
       		这两个实例分别在类级别和方法级别处理。
       		而annotation-driven配置帮助我们自动完成上述两个实例的注入。
    -->
	<mvc:annotation-driven/>

	<!-- 配置视图解析器: 
		处理请求的最后一件事情就是 "渲染输出" 控制器做出响应最后会经过这里进行渲染输出;
		DispatcherServlet(前端控制器)
			会查到一个视图解析器,将控制器返回的逻辑视图名称,渲染为一个指定的 实际视图文件上;
		Spring同样提供了多种...这里使用:InternalResourceViewResolver
	 -->
	 
	<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" >
		<!-- <property name="prefix" value="前缀"></property>   注意前缀前后加  / 分隔,方便文件拼接URL --> 
		<property name="suffix" value=".jsp"></property>
	</bean>	
	
	<!-- 
		suffix		后缀,为视图ViewName 添加后缀;
		prefix		前缀,默认文件为根目录  localhost:8080/项目名/  前/后缀就是 localhost:8080/项目名 前缀/文件 后缀
		这里建议和前面控制器搭配学习...
	 -->
</beans>

HelloController.java

代码语言:javascript复制
package com.wsm.controller;

import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.servlet.ModelAndView;

@Controller   					//@Controller是为了让Spring IOC容器初始化时自动扫描到;
//@RequestMapping("/WSM")		//@RequestMapping 声明在类上;  (对应示例二,示例一注释即可..)
public class HelloController {
//示例一 
	 @RequestMapping("/Hollo1.html") 
	public ModelAndView Hollo(String name){				//方法参数 name 和表单name 一样可以自动匹配,如果不一样会返回null; 
		//控制台打印输出显示								//所以一定要注意参数名一致哦!
		System.out.println("进入Hello:Controller");
		
		name = "Hello:" name;							//参数拼接
		//创建一个 ModelAndView 对象;
		ModelAndView mav = new ModelAndView();	
		mav.addObject("msg",name);						//存储
		mav.setViewName("index");						//返回响应
		return mav;
	}
	 
	 
	 
// 这里开始使用注解修改Demo

 /*
 *	首先这里的控制器和之前已经不一样了,而是直接的一个类; 并没有继承AbstractController类来实现 controller;
  *@Controller 
  *		而是使用了 @Controller 注解来使一个 JavaBean 成为一个controller(控制器);
  * 	类也没有重写什么方法,而是在类中定义方法,声明注解 @RequestMapping 来指定对应的控制器代码操作;
  * 	这样一方面节省了很多 控制器的类型声明,还避免了不同操作不同控制器代码的 op 验证操作;(直接根据注解找到对应的方法执行的代码,真好~)
  *@RequestMapping 
  *		用来指定控制器,完成映射,页面发起请求URL : localhost:8080/项目名 根目录截取掉 剩下的来这儿匹配; 找到对应的控制器;
  *		写法:
  *			@RequestMapping("/映射名")   或     @RequestMapping(value="/映射名") 
  * 	属性:
  * 		value="请求URL的匹配值"		method=method=RequestMethod.GET/POST 指定控制器响应的提交方式,默认都支持;
  *		@RequestMapping 可声明在: 方法 和 类 上面示例如下;
  *@RequestParam    
  *		当方法参数与 URL name相同时自动匹配值; 不同则默认null; 
  *		@RequestParam 就是当参数与  URL name不匹进行映射匹配的操作;
  *		属性:
  *			value="URL的name" 		required=false/true 默认true:URL中必须存在对应name进行匹配,不然参数异常 浏览器400!!为了解决建议设为 false; 
  *			defaultValue:设置默认属性值,如果没有指定,name匹配则使用默认值...eg: @RequestParam(defaultValue = "1")Integer dye 当前页默认第一页;
  *	扩展:
  *		(防止忘记,URL是以 name-value 且都是字符串类型)  localhost:8080/项目名/xx?name=value.. 存储方式(get显示post隐式存储)
  *		如果 GET/POST 存在相同Name 都会存在name中 ,逗号分隔;(请看index.jsp注释!) 
  */	 
//示例二
	 @RequestMapping(value="/Hollo2.html")  			
	 //可以看到 类/方法上各有一个注解; 使用时 URL: localhost:8080/项目名/WSM/Hollo2.html 指向该控制器; 方便分类操作,不同的控制器类针对不同的增删改查.. 
	public String Hollo2(@RequestParam(value="name",required=false) String na,Model model){													
		na = "Hello:" na;								//参数拼接
		//创建一个 ModelAndView 对象;
		ModelAndView mav = new ModelAndView();	
		mav.addObject("msg",na);						//这里方法返回的是String 而不是 ModelAndView 所以存储在这里的数据,页面并不会在接收到了...
		//参数对象 model
		model.addAttribute("msg",na);					//将数据存储在 model中,返回给页面接收;
		return "forward:../index.jsp";					//因为在类上加:@RequestMapping("/WSM")	会在根目录上默认加上 /WSM 所以需要 ../返回上一级目录;
		// 可以使用  return "redirect:页面" 或 return "forward:页面" 来完成 重定向/转发...
		// return 指定转发/重定向 ,跳转页面需要  文件后缀名,它不会在经过,前端控制器; 进行后缀的添加; 而是直接进行了重定向/转发;
		// 也不一定是非要带上 后缀。
		// 重定向就相当于是重新请求了一次:当前目录   "重定向的页面:xxxx";  URL地址发送变化;
		// 转发  就相当于在当前目录下请求:当前目录   "转发  的页面:xxxx";  URL地址不发生变化;
		// 经过web.xml 的DispatcherServlet 因为 / 所以 .jsp 会不在经过控制器直接去找页面. 如果不加 .jsp 还是会回到 控制中走~一遍 (亲测~)
	 }	 
	 // Model对象方法;
	 //	addAttribute(String,Object);		通过 key-value 方式来存储数值,默认存储request 中; 	
	 // addAttribute(Object); 				没有指定 key 直接存储value 方法会默认根据对象类型来作为 key; eg: String类型 key默认为 string 页面通过 ${string} 取值;
}

index.jsp

代码语言:javascript复制
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
<%
String path = request.getContextPath();
String basePath = request.getScheme() "://" request.getServerName() ":" request.getServerPort() path "/";
%>

<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
  <head>
    <base href="<%=basePath%>"> 
    
    <title>My JSP 'index.jsp' starting page</title>
	<meta http-equiv="pragma" content="no-cache">
	<meta http-equiv="cache-control" content="no-cache">
	<meta http-equiv="expires" content="0">    
	<meta http-equiv="keywords" content="keyword1,keyword2,keyword3">
	<meta http-equiv="description" content="This is my page">
	<!--
	<link rel="stylesheet" type="text/css" href="styles.css">
	-->
  </head>
  
  <body>
     <h2>${msg }</h2>   
     <!-- 如果表单提交中 action url中带参数 和 post方式提交的重名name-value 获取时候会同时获取 get,post表单 
     	    修改action 中连接(方式一):Hollo1.html?name=wsm 提交查看;
     -->
	<form action="Hollo1.html" method="post">   <!-- 使用实例二action 改为:WSM/Hollo2.html -->
		<input type="text" name="name">
		<input type="submit" value="提交">
	</form>
  </body>
</html>

参数传递(Controller to View)

1.ModelAndView

方法返回值为一个 ModelAndView 类型对象; 就像实例一通过对应的方法, 进行传参, 视图名称… 至前端控制器——视图解析器… 最终返回浏览器

2.Model

方法返回值是String 参数 Model 类型 (就像实例二一样)

3.Map

方法返回值是String 参数 Map< String,Object > 类型 实例中没有就是和 实例二类似, 参数为 Map< String,Object > ; 方法中通过 Map 对象.put 存储key-value

解释:

SpringMVC 的控制器的处理方法中 如果有 Map或Model 参数, 就会将请求内的 “隐含模型对象” 传递给这些形参, 因此可以通过 Map 和 Model 形参对模型中数据进行读写操作, (个人比较喜欢使用Model) 隐藏模型: SpringMVC 在调用方法前会 创建出一个隐含的模型对象,作为模型的存储容器; 如果传入参数为 Model ,SpringMVC 会将隐含模型传递给这些 参数存储; 开发者可以通过 参数访问到模型中的所有数据,当然也可往模型中新增属性数据。。。。

感谢观看-_-

0 人点赞