Spring MVC

MVC是什么?

    模型-视图-控制器(MVC)是一个以设计界面应用程序为基础的设计思想.是将业务逻辑、数据、显示分离的方法来组织代码。 MVC主要作用是降低了视图与业务逻辑间的双向偶

MVC的特点与优势 

      MVC的特点
        1.轻量级,简单易学
        2.高效 , 基于请求响应的MVC框架
        3.与Spring兼容性好,无缝结合
        4.约定优于配置 
        5.功能强大:RESTful、数据验证、格式化、本地化、主题等
        5.简洁灵活 


    MVC的优势

        1. 清晰的角色划分
        2. 分工明确,而且扩展点相当灵活,
        3. 和Spring 其他框架无缝集成,是其它Web框架所不具备的;
        4. 可适配,通过HandlerAdapter可以支持任意的类作为处理器;
        5. 可定制性,HandlerMapping、ViewResolver等能够非常简单的定制;
        6. 功能强大的数据验证、格式化、绑定机制;
        7. 利用Spring提供的Mock对象能够非常简单的进行Web层单元测试;
        8. 本地化、主题的解析的支持,使我们更容易进行国际化和主题的切换。
        9. 强大的JSP标签库,使JSP编写更容易。

SpringMVC 请求流程


   1. 发起请求到前端控制器(DispatcherServlet)
   2. 前端控制器请求HandlerMapping查找 Handler,可以根据xml配置、注解进行查找
   3. 处理器映射器HandlerMapping向前端控制器返回Handler
    4.前端控制器调用处理器适配器去执行Handler
   5. 处理器适配器去执行Handler
    6.Handler执行完成给适配器返回ModelAndView
    7.处理器适配器向前端控制器返回ModelAndView,ModelAndView是springmvc框架的一个底层对象,包括 Model和view
   8. 前端控制器请求视图解析器去进行视图解析,根据逻辑视图名解析成真正的视图(jsp)
    9.视图解析器向前端控制器返回View
   10. 前端控制器进行视图渲染,视图渲染将模型数据(在ModelAndView对象中)填充到request域
   11. 前端控制器向用户响应结果

 Spring MVC 环境搭建


    1. 开发环境
        Idea + Maven + Jdk1.8 + Jetty
    2. 新建 Maven webApp
    3. pom.xml 坐标添加

<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<maven.compiler.source>11</maven.compiler.source>
<maven.compiler.target>11</maven.compiler.target>
</properties>
<dependencies>
<!-- spring web -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-web</artifactId>
<version>5.2.4.RELEASE</version>
</dependency>
<!-- spring mvc -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-webmvc</artifactId>
<version>5.2.4.RELEASE</version>
</dependency>
<!-- web servlet -->
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>javax.servlet-api</artifactId>
<version>3.0.1</version>
<scope>provided</scope>
</dependency>
</dependencies>
<build>
<plugins>
<!-- 编译环境插件 -->
5.4. 配置 web.xml
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>2.3.2</version>
<configuration>
<source>11</source>
<target>11</target>
<encoding>UTF-8</encoding>
</configuration>
</plugin>
<!-- jetty插件 -->
<plugin>
<groupId>org.eclipse.jetty</groupId>
<artifactId>jetty-maven-plugin</artifactId>
<version>9.4.27.v20200227</version>
<configuration>
<scanIntervalSeconds>10</scanIntervalSeconds>
<!-- 设置端口 -->
<httpConnector>
<port>8080</port>
</httpConnector>
<!-- 设置项目路径 -->
<webAppConfig>
<contextPath>/springmvc01</contextPath>
</webAppConfig>
</configuration>
</plugin>
</plugins>
</build>


    4. 配置 web.xml

<?xml version="1.0" encoding="UTF-8"?>
<web-app id="WebApp_ID" version="3.0"
xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd">
<!-- 编码过滤 utf-8 -->
<filter>
<description>char encoding filter</description>
<filter-name>encodingFilter</filter-name>
<filterclass>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
<init-param>
<param-name>encoding</param-name>
<param-value>UTF-8</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>encodingFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<!-- servlet请求分发器 -->
<servlet>
要想启动我们的 SpringMVC 环境,目前对于 mvc 框架的配置还未进行。以上在 web.xml 中引用了
spring.xml 文件。
5.4.1. spring.xml 配置
<servlet-name>springMvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servletclass>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:spring.xml</param-value>
</init-param>
<!-- 表示启动容器时初始化该Servlet -->
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>springMvc</servlet-name>
<!-- 这是拦截请求, "/"代表拦截所有请求,"*.do"拦截所有.do请求 -->
<!-- <url-pattern>/</url-pattern> -->
<url-pattern>*.do</url-pattern>
</servlet-mapping>
</web-app>


        1. spring.xml 配置

<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc.xsd
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">
<!-- 开启扫描器 -->
<context:component-scan base-package="com.xxxx.springmvc.controller"/>
<!-- 使用默认的 Servlet 来响应静态文件 -->
<mvc:default-servlet-handler/>
<!-- 开启注解驱动-->
<mvc:annotation-driven/>
<!-- 配置视图解析器 -->
<bean
class="org.springframework.web.servlet.view.InternalResourceViewResolver"
id="internalResourceViewResolver">
<!-- 前缀:在WEB-INF目录下的jsp目录下 -->
<property name="prefix" value="/WEB-INF/jsp/"/>
<!-- 后缀:以.jsp结尾的资源 -->
<property name="suffix" value=".jsp"/>
</bean>
</beans>


        2. 页面控制器的编写

@Controller
public class HelloController {
/**
* 请求映射地址 /hello.do
* @return
*/
@RequestMapping("/hello")
public ModelAndView hello(){
ModelAndView mv=new ModelAndView();
mv.addObject("hello", "hello spring mvc");
mv.setViewName("hello");
return mv;
}
}


        3. 添加视图页面
            在 WEB-INF 下新建 jsp 文件夹 ,并在文件夹下新建 hello.jsp

<%@page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%
String path = request.getContextPath();
String basePath =
request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+pa
th+"/";
%>
<!DOCTYPE HTML>
<html>
<head>
<base href="<%=basePath %>">
<title>My JSP 'hello.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">
</head>
<body>
<!-- el表达式接收参数值 -->
${hello}
</body>
</html>


        4. 启动 jetty 服务器

URL 地址映射配置


   通过注解 @RequestMapping 将请求地址与方法进行绑定,可以在类级别和方法级别声明。类级别的 注解负责将一个特定的请求路径映射到一个控制器上,将 url 和类绑定;通过方法级别的注解可以细化 映射,能够将一个特定的请求路径映射到某个具体的方法上,将 url 和类的方法绑定。
    1. 映射单个 URL
        @RequestMapping("") 或 @RequestMapping(value="")
        @RequestMapping 声明在方法上面,映射单个 URL.路径开头是否加 斜杠"/" 均可

/**
* @RequestMapping 声明在方法上面,映射单个 URL
* 访问地址:(如果有类路径需要写在方法路径前面)
* http://ip:port/springmvc01/test01
* @return
*/
@RequestMapping("/test01")
// @RequestMapping(value = "/test01")
public ModelAndView test01(){
ModelAndView modelAndView = new ModelAndView();
modelAndView.addObject("hello","test01");
modelAndView.setViewName("hello");
return modelAndView;
}
/**
* 路径开头是否加 斜杠"/" 均可
* @RequestMapping("/请求路径") 与 @RequestMapping("请求路径")均可
* 建议加上,如:@RequestMapping("/test02")
* 访问地址:(如果有类路径需要写在方法路径前面)
* http://ip:port/springmvc01/test02
6.1.2. 映射多个 URL
@RequestMapping({"",""}) 或 @RequestMapping(value={"",""})
6.1.3. 映射 URL 在控制器上
用于类上,表示类中的所有响应请求的方法都是以该地址作为父路径。
* @return
*/
@RequestMapping("test02")
public ModelAndView test02(){
ModelAndView modelAndView = new ModelAndView();
modelAndView.addObject("hello","test02");
modelAndView.setViewName("hello");
return modelAndView;
}


    2. 映射多个 URL
        @RequestMapping({"",""}) 或 @RequestMapping(value={"",""})
        * @RequestMapping 声明在方法上面,映射多个 URL 支持一个方法绑定多个 url 的操作

@RequestMapping({"",""}) 或 @RequestMapping(value={"",""})
6.1.3. 映射 URL 在控制器上
用于类上,表示类中的所有响应请求的方法都是以该地址作为父路径。
* @return
*/
@RequestMapping("test02")
public ModelAndView test02(){
ModelAndView modelAndView = new ModelAndView();
modelAndView.addObject("hello","test02");
modelAndView.setViewName("hello");
return modelAndView;
}
/**
* @RequestMapping 声明在方法上面,映射多个 URL
* 支持一个方法绑定多个 url 的操作
* 访问地址:(如果有类路径需要写在方法路径前面)
* http://ip:port/springmvc01/test03_01
* http://ip:port/springmvc01/test03_02
* @return
*/
@RequestMapping({"/test03_01","/test03_02"})
// @RequestMapping(value = {"/test03_01","/test03_02"})
public ModelAndView test03(){
ModelAndView modelAndView = new ModelAndView();
modelAndView.addObject("hello","test03");
modelAndView.setViewName("hello");
return modelAndView;
}


    3. 映射 URL 在控制器上
        用于类上,表示类中的所有响应请求的方法都是以该地址作为父路径。

@Controller
@RequestMapping("/url")
public class UrlController {
/**
* @RequestMapping 声明在类上面,类中的的方法都是以该地址作为父路径
* 声明级别:
* 类级别 + 方法级别 (/类路径/方法路径)
* 访问地址:
* http://ip:port/springmvc01/url/test04
* @return
*/
@RequestMapping("/test04")
public ModelAndView test04(){
ModelAndView modelAndView = new ModelAndView();
modelAndView.addObject("hello","test04");
modelAndView.setViewName("hello");
return modelAndView;
}
}

4. 设置 URL 映射的请求方式

        默认没有设置请求方式,在HTTP 请求中最常用的请求方法是 GET、POST,还有其他的一些方法, 如:DELETE、PUT、HEAD 等。

        可以通过 method 属性设置支持的请求方式,如 method=RequestMethod.POST;如设置多种请求 方式,以大括号包围,逗号隔开即可。

/**
* 设置请求方式
* 通过 method 属性设置方法支持的请求方式,默认 GET请求和 POST等请求都支持。
* 设置了请求方式,则只能按照指定的请求方式请求。
* 访问地址:(只能使用POST请求访问)
* http://ip:port/springmvc01/url/test05
* @return
*/
@RequestMapping(value = "/test05",method = RequestMethod.POST)
public ModelAndView test05(){
ModelAndView modelAndView = new ModelAndView();
modelAndView.addObject("hello","test05");
modelAndView.setViewName("hello");
return modelAndView;
}

5. 通过参数名称映射 URL

/**
* 通过参数名称访问
* 通过参数的形式访问
* 访问地址:
* http://ip:port/springmvc01/url?test06
* @return
*/
@RequestMapping(params = "test06")
public ModelAndView test06(){
ModelAndView modelAndView = new ModelAndView();
modelAndView.addObject("hello","test06");
modelAndView.setViewName("hello");
return modelAndView;
}

参数绑定


    1. 基本数据类型
        通过注解 @RequestParam 标记一个形参为请求参数。
        参数值必须存在。如果没有指定参数值,也没有设置参数默认值,则会报500异常。
        设置参数的默认值 defaultValue
        设置参数的参数名(别名) name
    .2. 包装类型
        客户端请求参数名与方法形参名保持一致,默认参数值为null
        可以通过 @RequestParam 的name属性设置参数的别名,defaultValue属性设置参数默认值
    3. 字符串类型
        可以通过 @RequestParam 的name属性设置参数的别名,defaultValue属性设置参数默认值
    4. 数组类型
    5. JavaBean 类型
        客户端请求的参数名与JavaBean对象的属性字段名保持一致
    6. List 类型
        此时 User 实体需要定义对应 list 属性。(对于集合的参数绑定,一般需要使用 JavaBean 对象进行包装
    7.Set 类型
    8. Map 类型
        Map最为灵活,它也需要绑定在对象上,而不能直接写在Controller方法的参数中。

 请求转发与重定向

    1.重定向 redirect:
        重定向是发一个302的状态码给浏览器,浏览器自己去请求跳转的网页。地址栏会发生改变。
        重定向以 redirect: 开头
            return "redirect:view.jsp";
        传递参数
            return "redirect:view.jsp?uname=zhangsan&upwd=123456"

/**
* 重定向到JSP页面
* @return
*/
@RequestMapping(value="/view01")
public String view01(){
return "redirect:view.jsp";
}
/**
* 重定向到JSP页面
* 传递参数
* @return
*/
@RequestMapping(value="/view02")
public String view02(){
return "redirect:view.jsp?uname=zhangsan&upwd=123456";
}
/**
* 重定向到JSP页面
* 传递参数 (传递中文参数会出现乱码)
* @return
*/
@RequestMapping(value="/view03")
public String view03(){
return "redirect:view.jsp?uname=张三&upwd=123456";
}
/**
* 重定向到JSP页面
* 传递参数 (通过 RedirectAttributes 对象设置重定向参数,避免中文乱码问题)
* @param redirectAttributes
* @return
*/
@RequestMapping(value="/view04")
public String view04(RedirectAttributes redirectAttributes){
redirectAttributes.addAttribute("uname","张三");
redirectAttributes.addAttribute("upwd","123456");
return "redirect:view.jsp";
}
/**
* 重定向到JSP页面
* 返回 ModelAndView 对象
* @param modelAndView
* @return
*/
@RequestMapping(value="/view06")
public ModelAndView view06(ModelAndView modelAndView){
modelAndView.addObject("uname","李四");
modelAndView.addObject("upwd","123321");
modelAndView.setViewName("redirect:view.jsp");
return modelAndView;
}
/**
* 重定向到Controller
* 返回 ModelAndView 对象
* @param modelAndView
页面中获取参数值
8.2. 请求转发
请求转发,直接调用跳转的页面,让它返回。对于浏览器来说,它无法感觉服务器有没有forward。
地址栏不发生改变。可以获取请求域中的数据。
请求转发以 forward: 开头
* @return
*/
@RequestMapping(value="/view07")
public ModelAndView view07(ModelAndView modelAndView){
modelAndView.addObject("uname","admin");
modelAndView.setViewName("redirect:test01");
return modelAndView;
}


    2. 请求转发forward:
        地址栏不发生改变。可以获取请求域中的数据。
        请求转发以 forward: 开头
            return "forward:view.jsp";
        设置参数
            return "forward:view.jsp?uname=张三&upwd=123456";

/**
* 请求转发到JSP页面
*/
@RequestMapping("/view08")
public String view08(){
return "forward:view.jsp";
}
/**
* 请求转发到JSP页面
* 设置参数
*/
@RequestMapping("/view09")
public String view09(){
return "forward:view.jsp?uname=张三&upwd=123456";
}
/**
* 请求转发到JSP页面
* 设置请求域
*/
@RequestMapping("/view10")
public String view10(Model model){
model.addAttribute("uname","张三");
return "forward:view.jsp";
}
/**
* 请求转发到JSP页面 (默认)
* 默认会去指定目录下找JSP页面 (配置文件中设置的)
*/
@RequestMapping("/view11")
public String view11(){
页面中获取数据
9. JSON 数据开发
9.1. 基本概念
Json 在企业开发中已经作为通用的接口参数类型,在页面(客户端)解析很方便。SpringMVC 对于
json 提供了良好的支持,这里需要修改相关配置,添加 json 数据支持功能
9.1.1. @ResponseBody
该注解用于将 Controller 的方法返回的对象,通过适当的 HttpMessageConverter 转换为指定格式
后,写入到 Response 对象的 body 数据区。
返回的数据不是 html 标签的页面,而是其他某种格式的数据时(如 json、xml 等)使用(通常用于
ajax 请求)。
9.1.2. @RequestBody
该注解用于读取 Request 请求的 body 部分数据,使用系统默认配置的 HttpMessageConverter 进行
解析,然后把相应的数据绑定到要返回的对象上 ,再把 HttpMessageConverter 返回的对象数据绑定到
controller 中方法的参数上。
return "/../../view";
}
/**
* 请求转发到 Controller
* @return
*/
@RequestMapping("/view12")
public ModelAndView view12(ModelAndView modelAndView){
modelAndView.setViewName("forward:test01");
return modelAndView;
}
/**
* 请求转发到 Controller
* 传递参数
* @return
*/
@RequestMapping("/view13")
public ModelAndView view13(ModelAndView modelAndView){
modelAndView.setViewName("forward:test01?uname=admin");
return modelAndView;
}

JSON 数据开发

        作为通用的接口参数类型,在页面(客户端)解析很方便,需要修改相关配置,添加 json 数据支持功能

    .1. @ResponseBody
        通过@requestBody注解实现将json数据转成java对象。内部使用jackson包将json转成java对象。

    2. @RequestBody
        通过@responseBody注解实现将java对象转成json输出
    使用配置
        .1. 添加 json相关坐标

<!-- 添加json 依赖jar包 -->
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-core</artifactId>
<version>2.10.0</version>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-databind</artifactId>
<version>2.10.0</version>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-annotations</artifactId>
<version>2.10.0</version>
</dependency>


            
        2. 修改配置文件
            mvc 请求映射 处理器与适配器配置

<!-- mvc 请求映射 处理器与适配器配置 -->
<mvc:annotation-driven>
<mvc:message-converters>
<bean
class="org.springframework.http.converter.StringHttpMessageConverter" />
<bean
class="org.springframework.http.converter.json.MappingJackson2HttpMessageConvert
er" />
</mvc:message-converters>
</mvc:annotation-driven>


    . 注解使用
        @ResponseBody
            注解设置在方法体上
            注解设置在方法返回对象前,修饰符之后

@Controller
@RequestMapping("/user")
public class UserController {
/**
* @ResponseBody 返回的是JOSN格式的数据,返回JavaBean对象
* 注解设置在方法体上
* @return
*/
@RequestMapping("queryUser01")
@ResponseBody
public User queryUser01(){
User user = new User();
user.setId(1);
9.2.3.2. @RequestBody
@RequestBody 注解常用来处理 content-type 不是默认的 application/x-www-form-urlcoded 类型
的内容,比如说:application/json 或者是application/xml 等。一般情况下来说常用其来处理
application/json 类型。@RequestBody接受的是一个 json 格式的字符串,一定是一个字符串。
通过 @RequestBody 可以将请求体中的 JSON 字符串绑定到相应的 bean 上,当然,也可以将其分别
绑定到对应的字符串上。
user.setUserName("zhangsan");
user.setUserPwd("123456");
// 返回的是user对象
return user;
}
/**
* @ResponseBody 返回的是JOSN格式的数据,返回JavaBean对象
* 注解设置在方法返回对象前,修饰符之后
* @return
*/
@RequestMapping("queryUser02")
public @ResponseBody User queryUser02(){
User user = new User();
user.setId(2);
user.setUserName("lisi");
user.setUserPwd("123321");
// 返回的是user对象
return user;
}
/**
* @ResponseBody 返回的是JOSN格式的数据,返回集合
* @return
*/
@RequestMapping("/queryUser03")
@ResponseBody
public List<User> queryUser03(){
List<User> list = new ArrayList<>();
User user01 = new User();
user01.setId(1);
user01.setUserName("zhangsan");
user01.setUserPwd("123456");
User user02 = new User();
user02.setId(2);
user02.setUserName("lisi");
user02.setUserPwd("123321");
list.add(user01);
list.add(user02);
// 返回的是user集合
return list;
}
}


         @RequestBody
            接受的是一个 json 格式的字符串,一定是一个字符串
            注解设置在形参前面
 

/**
* @RequestBody 规定请求的参数是JOSN格式的字符串
* 注解设置在形参前面
* @param user
* @return
*/
@RequestMapping("/getUser")
@ResponseBody
public User getUser(@RequestBody User user){
System.out.println(user);
return user;
}

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>JSON处理</title>
<%-- 引入Jquery的核心JS文件 --%>
<script type="text/javascript" src="js/jquery-3.4.1.js"></script>
</head>
<body>
<input type="button" value="JSON数据测试" onclick="test()"/>
<script type="text/javascript">
/**
* 请求传递JSON格式的数据
* 返回JSON格式的数据
*/
function test(){
$.ajax({
// 请求方式 Get|Post
type: "post",
// 请求路径
url: "user/getUser",
// 预期服务器返回的额数据类型
dataType: "json",
// 设置服务器请求类型的数据类型为JSON格式
contentType: "application/json;charset=utf-8",
// 传递给服务器的参数
data:‘{"userName":"admin","userPwd":"123456"}’,
// 回调函数,接收服务器返回的响应的结果 (函数中的形参用来接收服务器返回的数
据)
success:function(data){
console.log(data);
}
})
}
</script>
</body>
</html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值