Java(SpringMVC01)
参考视频:SpringMVC最新教程IDEA版通俗易懂(狂神)
1. 前言
- SSM:Mybatis + Spring + SpringMVC( MVC三层架构 )
- JavaSE:认真学习,老师带,入门快。
- JavaWeb:认真学习,老师带,入门快。
- SSM 框架 :研究官方文档,锻炼笔记能力,锻炼项目能力。
- 后面路线:SpringMVC + Vue + SpringBoot + SpringCloud + Linux…
- SSM框架 = JavaWeb,一直做项目,熟练搭建…
- Spring关键点:IOC 和 AOP
- SpringMVC关键点理论:SpringMVC的执行流程。
- SpringMVC关键点实践:SSM框架整合(没有博客搭不起来…)(多练…)
2. 什么是MVC(回顾)
- MVC是模型(Model)、视图(View)、控制器(Controller)的简称,是一种软件设计规范
- 是将业务逻辑、数据、显示分离的方法来组织代码
- MVC主要作用是 降低了视图与业务逻辑间的双向耦合。
- MVC不是一种设计模式,MVC是一种架构模式。 当然不同的MVC存在差异。
- Model(模型)层: 数据模型,提供要展示的数据,因此包含数据和行为,可以认为是领域模型或JavaBean组件(包含数据和行为),不过现在一般都分离开来:Value Object(数据Dao) 和 服务层(行为Service)。也就是模型提供了模型数据查询和模型数据的状态更新等功能,包括数据和业务。
- View(视图)层: 负责进行模型的展示,一般就是我们见到的用户界面,客户想看到的东西。
- Controller(控制器)层: 接收用户请求,委托给模型进行处理(状态改变),处理完毕后把返回的模型数据返回给视图,由视图负责展示。也就是说控制器做了个调度员的工作。
- 最典型的MVC就是JSP + servlet + javabean的模式。
2.1 Model1时代
- 在web早期的开发中,通常采用的都是Model1。
- Model1中,主要分为两层,视图层和模型层。
- Model1优点:
- 架构简单,比较适合小型项目开发。
- Model1缺点:
- JSP职责不单一,职责过重,不便于维护。
2.2 Model2时代
- Model2把一个项目分成三部分,包括 视图、控制、模型。
- 用户发请求
- Servlet接收请求数据,并调用对应的业务逻辑方法
- 业务处理完毕,返回更新后的数据给servlet
- servlet转向到JSP,由JSP来渲染页面
- 响应给前端更新后的页面
- 职责分析:
- Controller:控制器
- 取得表单数据
- 调用业务逻辑
- 转向指定的页面
- Model:模型
- 业务逻辑
- 保存数据的状态
- View:视图
- 显示页面
- Controller:控制器
2.3 杂谈
- dao
- service
- servlet :转发,重定向
- jsp/html
- 前端、数据传输、实体类
- 实体类:用户名、密码、生日、爱好…(20个)
- 前端:用户名、密码(2个)
- pojo:User(20个)
- vo:UserVo(2个)
- dto:…
- JSP:本质就是一个Servlet!(现在基本都不用jsp了…)
- 假设:你的项目的架构,是设计好的,还是 演进 的?
- Alibaba-----PHP
- 随着用户量增大-----Java
- 王坚-----去IOE-----MySQL
- MySQL:MySQL-----> AliSQL、AliRedis
- All in one-----> 微服务
3. 回顾Servlet
- 新建工程(普通Maven)-----删除src目录(用不到)-----导包(依赖)
<dependencies>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-webmvc</artifactId>
<version>5.1.9.RELEASE</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>servlet-api</artifactId>
<version>2.5</version>
</dependency>
<dependency>
<groupId>javax.servlet.jsp</groupId>
<artifactId>jsp-api</artifactId>
<version>2.2</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>jstl</artifactId>
<version>1.2</version>
</dependency>
</dependencies>
- 新建子Module-----springmvc-01-servlet
- 现在创建子Module时,parent标签(不知为何)不会因为IDEA的bug自动删除了…
- 右键 子项目 -----Add Framework Support-----勾选WebApplication(4.0)
- 可以在子项目中导入下面的依赖(不用应该也可以)
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>servlet-api</artifactId>
<version>2.5</version>
</dependency>
<dependency>
<groupId>javax.servlet.jsp</groupId>
<artifactId>jsp-api</artifactId>
<version>2.2</version>
</dependency>
- 编写一个Servlet类,用来处理用户的请求
package com.zach.servlet;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
public class HelloServlet extends HttpServlet {
// 全栈:后台 + 前端 + 数据库 + 运维
// 前端:后台 + 前端
// Python != 人工智能...
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//1. 获取前端参数
String method = req.getParameter("method");
if (method.equals("add")) {
req.getSession().setAttribute("msg","执行了add方法");
}
if (method.equals("delete")){
req.getSession().setAttribute("msg","执行了delete方法");
}
//2. 调用业务层
//3. 视图转发 或者 重定向
req.getRequestDispatcher("/WEB-INF/jsp/test.jsp").forward(req,resp);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//用doPost调用doGet,可以实现复用。
doGet(req, resp);
}
}
- jsp页面
- form.jsp
<body>
<form action="/hello" method="post">
<input type="text" name="method">
<input type="submit"/>
</form>
</body>
- test.jsp
<body>
${msg}
</body>
- web.xml
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
version="4.0">
<servlet>
<servlet-name>HelloServlet</servlet-name>
<servlet-class>com.zach.servlet.HelloServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>HelloServlet</servlet-name>
<url-pattern>/hello</url-pattern>
</servlet-mapping>
<!--15分钟session失效-->
<!-- <session-config>-->
<!-- <session-timeout>15</session-timeout>-->
<!-- </session-config>-->
<!--欢迎页面:默认是index.jsp-->
<!-- <welcome-file-list>-->
<!-- <welcome-file>index.jsp</welcome-file>-->
<!-- </welcome-file-list>-->
</web-app>
- 配置Tomcat并启动测试
- http://localhost:8080/hello-----500错误
- http://localhost:8080/hello?method=delete-----执行了delete方法
- 其他
- MVC:M-----V-----C
- MVVM:M-----V-----VM(ViewModel:双向绑定)
4. 初识SpringMVC
4.1 什么是SpringMVC
- Spring MVC是Spring Framework的一部分,是基于Java实现MVC的轻量级Web框架。
- 查看官方文档:
- https://docs.spring.io/spring/docs/5.2.0.RELEASE/spring-framework-reference/web.html#spring-web
- 我们为什么要学习springmvc?
- Spring MVC的特点:
- 轻量级,简单易学
- 高效 , 基于请求响应的MVC框架
- 与Spring兼容性好,无缝结合
- 约定优于配置
- 功能强大:RESTful、数据验证、格式化、本地化、主题等
- 简洁灵活
- Spring MVC的特点:
- Spring的Web框架围绕 DispatcherServlet 【调度servlet】设计。
- DispatcherServlet的作用是将请求分发到不同的处理器。从Spring 2.5开始,使用Java 5或者以上版本的用户可以采用基于注解形式进行开发,十分简洁!
- 正因为SpringMVC好、简单、便捷、易学、天生和Spring无缝集成(使用SpringIoC和Aop),使用约定优于配置。能够进行简单的junit测试。支持Restful风格,异常处理,本地化,国际化,数据验证,类型转换,拦截器等等…所以我们要学习。
- 用的公司多。
4.2 中心控制器
- Spring的web框架围绕DispatcherServlet设计。DispatcherServlet的作用是将请求分分发到不同的处理器。从Spring2.5开始,开始java5或者以上版本的用户可以采用基于注解的controller声明方式。
- Spring MVC框架像许多其他MVC框架一样, 以请求为驱动 , 围绕一个中心Servlet分派请求及提供其他功能,DispatcherServlet是一个实际的Servlet (它继承自HttpServlet 基类)。
- DispatcherServlet类图如下:
- SpringMVC的原理如下图所示:
- 当发起请求时被前置的控制器拦截到请求,根据请求参数生成代理请求,找到请求对应的实际控制器,控制器处理请求,创建数据模型,访问数据库,将模型响应给中心控制器,控制器使用模型与视图渲染视图结果,将结果返回给中心控制器,再将结果返回给请求者。
4.3 SpringMVC执行原理
- 下图为SpringMVC的一个较完整的流程图,实线表示SpringMVC框架提供的技术,不需要开发者实现,虚线表示需要开发者实现。
简要分析执行流程
- DispatcherServlet表示 前置控制器 ,是整个SpringMVC的控制中心,用户发出请求,DispatcherServlet接收请求并拦截请求。
- 我们假设请求的url为 : http://localhost:8080/SpringMVC/hello
- 如上url拆分成三部分:
- http://localhost:8080 服务器域名
- SpringMVC部署在服务器上的web站点
- hello表示控制器
- 通过分析,如上url表示为:请求位于服务器localhost:8080上的SpringMVC站点的hello控制器。
-
HandlerMapping为 处理器映射器 。DispatcherServlet调用HandlerMapping,HandlerMapping根据请求Url找Handler
-
HandlerExecution表示具体的Handler,其主要作用是根据URL查找控制器,如上url被查找控制器为:hello
-
HandlerExecution将解析后的信息传递给DispatcherServlet,如解析控制器映射等
-
HandlerAdapter表示 处理器适配器 ,其按照特定的规则去执行Handler
-
Handler让具体的Controller执行
-
Controller将具体的执行信息返回给HandlerAdapter,如ModelAndView
-
HandlerAdapter将视图逻辑名或模型传递给DispatcherServlet
-
DispatcherServlet调用 视图解析器 (ViewResolver)来解析HandlerAdapter传递的逻辑视图名
-
视图解析器将解析的逻辑视图名传给DispatcherServlet
-
DispatcherServlet根据视图解析器解析的视图结果,调用具体的视图
-
最终视图呈现给用户
- 在这里先听一遍原理,不理解没有关系,我们马上来写一个对应的代码实现大家就明白了,如果不明白,那就写10遍,没有笨人,只有懒人!(?)
5. HelloSpringMVC
5.1 配置版
-
新建一个Module(springmvc-02-hellomvc)-----添加web的支持(Add Framework Support)
-
确定导入了SpringMVC的依赖
-
配置web.xml,注册DispatcherServlet
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
version="4.0">
<!--配置(注册)DispatcherServlet:这是SpringMVC的核心:请求分发器、前端(前置)控制器-->
<servlet>
<servlet-name>springmvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<!--关联一个springmvc的配置文件:格式:【servlet-name】-servlet.xml-->
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:springmvc-servlet.xml</param-value>
</init-param>
<!--启动级别-1-->
<load-on-startup>1</load-on-startup>
</servlet>
<!--在SpringMVC中,/和/*不同:
/ 匹配所有的请求;(但不包括.jsp)
/* 匹配所有的请求;(且包括.jsp)(写/*会无限循环...)
-->
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
</web-app>
- 编写SpringMVC 的 配置文件!名称:
springmvc-servlet.xml : [servletname]-servlet.xml
。这里的名称要求是按照官方来的。 - 添加 处理器映射器
- 添加 处理器适配器
- 添加 视图解析器
- springmvc-servlet.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
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">
<!--处理器映射器-->
<bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping"/>
<!--处理器适配器-->
<bean class="org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter"/>
<!--视图解析器:DispatcherServlet给他的ModelAndView-->
<!--
1. 获取了ModelAndView的数据
2. 解析ModelAndView视图的名字
3. 拼接视图名字,找到对应的视图 /WEB-INF/jsp/hello.jsp
4. 将数据渲染到这个视图上!
-->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" id="InternalResourceViewResolver">
<!--前缀-->
<property name="prefix" value="/WEB-INF/jsp/"/>
<!--后缀-->
<property name="suffix" value=".jsp"/>
</bean>
<!--Handler-->
<bean id="/hello" class="com.zach.controller.HelloController"/>
</beans>
- 编写我们要操作的业务Controller ,要么实现Controller接口,要么增加注解;需要返回一个ModelAndView,装数据,封视图
package com.zach.controller;
import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.mvc.Controller;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
//注意:这里我们先导入Controller接口
public class HelloController implements Controller {
public ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response) throws Exception {
//ModelAndView 模型和视图
ModelAndView mv = new ModelAndView();
//业务代码:
//调用业务层
String result = "HelloSpringMVC...";
//视图跳转:
//封装对象,放在ModelAndView中。Model
mv.addObject("msg", "HelloSpringMVC!!!");
//封装要跳转的视图,放在ModelAndView中
mv.setViewName("hello");//: /WEB-INF/jsp/hello.jsp
return mv;
}
}
- 将自己的类交给SpringIOC容器,注册bean
<!--Handler-->
<bean id="/hello" class="com.zach.controller.HelloController"/>
- 写要跳转的jsp页面,显示ModelandView存放的数据,以及我们的正常页面
<body>
${msg}
</body>
- 配置tocmat,启动测试
- 可能遇到的问题:访问出现404,排查步骤:
- 查看控制台输出,看一下是不是缺少了什么jar包。
- 如果jar包存在,显示无法输出,就在IDEA的项目发布中,添加lib依赖!
- 重启Tomcat 即可解决!
5.2 注解版
-
新建一个Moudle,springmvc-03-annotation 。添加web支持!
-
由于Maven可能存在资源过滤的问题,我们将配置完善
<!--在build中配置resources,来防止我们资源导出失败的问题-->
<build>
<resources>
<resource>
<directory>src/main/resources</directory>
<includes>
<include>**/*.properties</include>
<include>**/*.xml</include>
</includes>
<filtering>false</filtering>
</resource>
<resource>
<directory>src/main/java</directory>
<includes>
<include>**/*.properties</include>
<include>**/*.xml</include>
</includes>
<filtering>false</filtering>
</resource>
</resources>
</build>
-
在pom.xml文件引入相关的依赖:主要有Spring框架核心库、Spring MVC、servlet 、JSTL等。我们在父依赖中已经引入了!
-
配置web.xml
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
version="4.0">
<!--注册DispatcherServlet-->
<servlet>
<servlet-name>springmvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<!--通过初始化参数指定SpringMVC配置文件的位置,进行关联-->
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:springmvc-servlet.xml</param-value>
</init-param>
<!--启动顺序,数字越小,优先级越高,启动越早-->
<load-on-startup>1</load-on-startup>
</servlet>
<!--所有请求都会被springmvc拦截 -->
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
</web-app>
-
注意web.xml版本问题,要最新版!
-
注册DispatcherServlet
-
关联SpringMVC的配置文件
-
启动级别为1
-
映射路径为 / 【不要用/*,会404】
/ 和 /* 的区别:
- < url-pattern > / </ url-pattern > 不会匹配到.jsp, 只针对我们编写的请求;即:.jsp 不会进入spring的 DispatcherServlet类
- < url-pattern > /* </ url-pattern > 会匹配 *.jsp,会出现返回 jsp视图 时再次进入spring的DispatcherServlet 类,导致找不到对应的controller所以报404错
- 添加SpringMVC配置文件
- springmvc-servlet.xml
<?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:context="http://www.springframework.org/schema/context"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
https://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/mvc
https://www.springframework.org/schema/mvc/spring-mvc.xsd">
<!--自动扫描包,让指定包下的注解生效,有IOC容器统一管理-->
<context:component-scan base-package="com.zach.controller"/>
<!--让Spring MVC不处理静态资源 .css .js .html ...-->
<mvc:default-servlet-handler/>
<!--支持mvc注解驱动
在spring中一般采用@RequestMapping注解来完成映射关系
要想使@RequestMapping注解生效
必须向上下文中注册DefaultAnnotationHandlerMapping
和一个AnnotationMethodHandlerAdapter实例
这两个实例分别在类级别和方法级别处理。
而annotation-driven配置帮助我们自动完成上述两个实例的注入。
-->
<mvc:annotation-driven/>
<!--
视图解析器:DispatcherServlet给他的ModelAndView
1. 获取了ModelAndView的数据
2. 解析ModelAndView视图的名字
3. 拼接视图名字,找到对应的视图 /WEB-INF/jsp/hello.jsp
4. 将数据渲染到这个视图上!
-->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" id="internalResourceViewResolver">
<!--前缀-->
<property name="prefix" value="/WEB-INF/jsp/"/>
<!--后缀-->
<property name="suffix" value=".jsp"/>
</bean>
</beans>
- 在视图解析器中我们把所有的视图都存放在
/WEB-INF/
目录下,这样可以保证视图安全,因为这个目录下的文件,客户端不能直接访问。 - 让IOC的注解生效
- 静态资源过滤 :HTML、JS、CSS、图片、视频…
- MVC的注解驱动
- 配置视图解析器
- 创建Controller
package com.zach.controller;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestMapping;
@Controller
@RequestMapping("hello")
public class HelloController {
//此时访问路径:localhost:8080/hello/h1(一般类上不写,方法上写死,有助于排错)
@RequestMapping("/h1")
public String Hello(Model model){
//封装数据:向模型中添加属性msg与值,可以在JSP页面中取出并渲染
model.addAttribute("msg","Hello,SpringMVCAnnotation!!!");
return "hello"; //return说明会被视图解析器处理(?)。返回的是前后缀中间的字符串。WEB-INF/jsp/hello.jsp
}
}
- @Controller是为了让Spring IOC容器初始化时自动扫描到
- @RequestMapping是为了 映射请求路径 ,这里因为只有方法上有映射所以访问时应该是/hello
- 方法中声明Model类型的参数是为了把Action中的数据带到视图中
- 方法返回的结果是视图的名称hello,加上配置文件中的前后缀变成
WEB-INF/jsp/hello.jsp
- 创建视图层hello.jsp
- 在
WEB-INF/jsp
目录中创建hello.jsp ,视图可以直接取出并展示从Controller带回的信息 - 可以通过EL表示取出Model中存放的值,或者对象;
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
${msg}
</body>
</html>
- 配置Tomcat,运行测试
5.3 小结
实现步骤其实非常的简单:
- 新建一个web项目
- 导入相关jar包
- 编写web.xml,注册DispatcherServlet
- 编写springmvc配置文件
- 创建对应的控制类,controller
- 完善前端视图和controller之间的对应
- 测试运行调试
- 使用SpringMVC必须配置的三大件:
- 处理器映射器
- 处理器适配器
- 视图解析器
- 通常,我们只需要 手动配置视图解析器 ,而 处理器映射器 和 处理器适配器 只需要开启 注解驱动 即可,而省去了大段的xml配置。