Spring MVC

ssm:mybatis+Spring+SpringMVC MVC三层架构

JavaSE:认真学习,老师带,入门快

JavaWeb:认真学习,老师带,入门快

SSM框架:研究官方文档,锻炼自学能力,锻炼笔记能力,锻炼项目能力

Spring MVC + Vue + SpringBoot + SpringCloud + Linux

SSM=JavaWeb做项目;

Spring:IOC和AOP

SpringMVC:SpringMVC的执行流程!

SpringMVC:SSM框架整合!I

MVC:模型(dao,service) 视图(jsp) 控制器(Servlet)

dao

service

servlet:转发,重定向

jsp/html

前端 数据传输 实体类

实体类:用户名,密码,生日,爱好,…… 20个

前端:用户名,密码

pojo:User

vo:UserVo

dto:

Spring:大杂绘,我们可以将SpringMVC中所有要用到的bean,注册到Spring中!

什么是MVC?

  • MVC是模型(Model)、视图(View)、控制器(Controller)的简写,是一种软件设计规范。

  • 是将业务逻辑、数据、显示分离的方法来组织代码。

  • MVC主要作用是降低了视图与业务逻辑间的双向偶合

  • MVC不是一种设计模式,MVC是一种架构模式。当然不同的MVC存在差异。
    Model(模型):数据模型,提供要展示的数据,因此包含数据和行为,可以认为是领域模型或JavaBean组件(包含数据和行为),不过现在一般都分离开来:Value Object(数据Dao)和服务层(行为Service)。也就是模型提供了模型数据查询和模型数据的状态更新等功能,包括数据和业务。
    View(视图):负责进行模型的展示,一般就是我们见到的用户界面,客户想看到的东西。
    Controller(控制器):接收用户请求,委托给模型进行处理(状态改变),处理完毕后把返回的模型数据返回给视图,由视图负责展示。也就是说控制器做了个调度员的工作。

MVVM:M V V M ViewModel:双向绑定

  1. **最典型的MVC就是JSP+servlet+javabean的模式。**

Spring MVC - 图1

Model1时代

  • 在web早期的开发中,通常采用的都是Model1
  • Model1中,主要分为两层,视图层和模型层。

Spring MVC - 图2

Model1优点:架构简单,比较适合小型项目开发;

Model1缺点:JSP职责不单一,职责过重,不便于维护;

Model2时代

  1. Model2把一个项目分成三部分,包括**视图、控制、模型。**

Spring MVC - 图3

  1. 用户发请求
  2. Servlet接收请求数据,并调用对应的业务逻辑方法
  3. 业务处理完毕,返回更新后的数据给servlet
  4. servlet转向到JSP,由JSP来渲染页面
  5. 响应给前端更新后的页面

    职责分析:

    Controller:控制器

  6. 取得表单数据

  7. 调用业务逻辑
  8. 转向指定的页面

    Model:模型

  9. 业务逻辑

  10. 保存数据的状态

    View:视图

  11. 显示页面

    Model2这样不仅提高的代码的复用率与项目的扩展性,且大大降低了项目的维护成本。Model1模式的实现比较简单,适用于快速开发小规模项目,Model1中JSP页面身兼View和Controller两种角色,将控制逻辑和表现逻辑混杂在一起,从而导致代码的重用性非常低,增加了应用的扩展性和维护的难度。Model2消除了Model1的缺点。

回顾Servlet

  1. 新建一个Maven工程当作父工程!pom依赖
    1. <!--统一声明版本号-->
    2. <properties>
    3. <junit.version>4.12</junit.version>
    4. <spring-webmvc.version>5.1.9.RELEASE</spring-webmvc.version>
    5. <servlet-api.version>2.5</servlet-api.version>
    6. <jsp-api.version>2.2</jsp-api.version>
    7. <jstl.version>1.2</jstl.version>
    8. </properties>
    9. <!--导入依赖-->
    10. <dependencies>
    11. <!--测试类-->
    12. <dependency>
    13. <groupId>junit</groupId>
    14. <artifactId>junit</artifactId>
    15. <version>${junit.version}</version>
    16. </dependency>
    17. <!--MVC-->
    18. <dependency>
    19. <groupId>org.springframework</groupId>
    20. <artifactId>spring-webmvc</artifactId>
    21. <version>${spring-webmvc.version}</version>
    22. </dependency>
    23. <!--servlet-->
    24. <dependency>
    25. <groupId>javax.servlet</groupId>
    26. <artifactId>servlet-api</artifactId>
    27. <version>${servlet-api.version}</version>
    28. </dependency>
    29. <!--jsp-->
    30. <dependency>
    31. <groupId>javax.servlet.jsp</groupId>
    32. <artifactId>jsp-api</artifactId>
    33. <version>${jsp-api.version}</version>
    34. </dependency>
    35. <dependency>
    36. <groupId>javax.servlet</groupId>
    37. <artifactId>jstl</artifactId>
    38. <version>${jstl.version}</version>
    39. </dependency>
    40. </dependencies>
  1. 建立一个Moudle:springmvc-01-servlet,添加Web app的支持!

  2. 导入servlet和jsp的jar依赖

    1. <dependencies>
    2. <!--servlet-->
    3. <dependency>
    4. <groupId>javax.servlet</groupId>
    5. <artifactId>servlet-api</artifactId>
    6. <version>2.5</version>
    7. </dependency>
    8. <!--jsp-->
    9. <dependency>
    10. <groupId>javax.servlet.jsp</groupId>
    11. <artifactId>jsp-api</artifactId>
    12. <version>2.2</version>
    13. </dependency>
    14. </dependencies>
  1. 编写一个Servlet类,用来处理用户的请求

    1. //实现Servlet接口
    2. public class HelloServlet extends HttpServlet {
    3. @Override
    4. protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
    5. //1.获取前端参数
    6. //取得参数
    7. String method = req.getParameter("method");
    8. if (method.equals("add")) {
    9. req.getSession().setAttribute("msg","执行了add方法");
    10. }
    11. if (method.equals("delete")) {
    12. req.getSession().setAttribute("msg","执行了delete方法");
    13. }
    14. //2.调用业务层
    15. //3.视图转发或重定向
    16. //业务逻辑
    17. //视图跳转
    18. req.getRequestDispatcher("WEB-INF/jsp/test.jsp").forward(req,resp);
    19. }
    20. @Override
    21. protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
    22. doGet(req, resp);
    23. }
    24. }
  1. 创建jsp页面 ```jsp <%@ page contentType=”text/html;charset=UTF-8” language=”java” %>

${msg}

  1. 6.
  2. web.xml中注册Servlet
  3. ```xml
  4. <?xml version="1.0" encoding="UTF-8"?>
  5. <web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
  6. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  7. xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
  8. version="4.0">
  9. <servlet>
  10. <servlet-name>hello</servlet-name>
  11. <servlet-class>cn.dafran.servlet.HelloServlet</servlet-class>
  12. </servlet>
  13. <servlet-mapping>
  14. <servlet-name>hello</servlet-name>
  15. <url-pattern>/hello</url-pattern>
  16. </servlet-mapping>
  17. <!-- <session-config>
  18. <session-timeout>15</session-timeout>
  19. </session-config>
  20. 设置默认超时时间-->
  21. <!-- <welcome-file-list>
  22. <welcome-file>index.jsp</welcome-file>
  23. </welcome-file-list>
  24. 欢迎页-->
  25. </web-app>
  1. 配置Tomcat,并启动测试

    • localhost:8080/user?method=add
    • localhost:8080/user?method=delete

MVC框架要做哪些事情

  1. 将url映射到java类或java类的方法.

  2. 封装用户提交的数据.

  3. 处理请求—调用相关的业务处理—封装响应数据.

  4. 响应的数据进行渲染.jsp/html等表示层数据.
    说明:
    常见的服务器端MVC框架有:Struts、Spring MVC、ASP.NET MVC、Zend Framework、JSF;常见前端MVC框架:vue、angularjs、react、backbone;由MVC演化出了另外一些模式 如:MVP、MVVM等等…

初始Spring MVC

概述

Spring MVC是Spring Framework的一部分,是基于Java实现MVC的轻量级Web框架。

查看官方文档:官方文档5.2.0官方文档4.3.24

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

    Spring的web框架围绕DispatcherServlet[调度Servlet]设计。

    DispatcherServlet的作用是将请求分发到不同的处理器。从Spring 2.5开始,使用Java 5或者以上版本的用户可以采用基于注解形式进行开发,十分简洁;

    正因为SpringMVC好,简单,便捷,易学,天生和Spring无缝集成(使用SpringIoC和Aop),使用约定优于配置.能够进行简单的junit测试.支持Restful风格.异常处理,本地化,国际化,数据验证,类型转换,拦截器等等……所以要学习.

    用的人多,使用公司多

中心控制器

原来:每个Servlet都需要做重定向或者转发

Spring MVC - 图4

MVC:加层处理 做调度

Spring MVC - 图5

  1. Springweb框架围绕DispatcherServlet设计。DispatcherServlet的作用是将请求分发到不同的处理器。从Spring 2.5开始,使用Java 5或者以上版本的用户可以采用基于注解的controller声明方式。<br />
  2. Spring MVC框架像许多其他MVC框架一样,**以请求为驱动,围绕一个中心Servlet分派请求及提供其他功能,DispatcherServlet是一个实际的Servlet(它继承自HttpServlet基类)。**

Spring MVC - 图6

  1. **DispatcherServlet 继承ServletServlert起的是一样的作用。**
  2. SpringMVC的原理如下图所示:
  3. 当发起请求时被前置的控制器拦截到请求,根据请求参数生成代理请求,找到请求对应的实际控制器,控制器处理请求,创建数据模型,访问数据库,将模型响应给中心控制器,控制器使用模型与视图渲染视图结果,将结果返回给中心控制器,再将结果返回给请求者。

Spring MVC - 图7

SpringMVC执行原理

HelloSpringMVC

配置版

  1. 新建一个Moudle,添加web的支持!

  2. 确定导入了SpringMVC的依赖!

  3. 配置web.xml,注册DispatcherServlet

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
  5. version="4.0">
  6. <!--1.注册Dispatcherservlet-->
  7. <servlet>
  8. <servlet-name>springmvc</servlet-name>
  9. <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
  10. <!--关联一个springmvc的配置文件:【servlet-name】-servlet.xml-->
  11. <init-param>
  12. <param-name>contextconfigLocation</param-name>
  13. <param-value>classpath:springmvc-servlet.xml</param-value>
  14. </init-param>
  15. <!--启动级别-1-->
  16. <load-on-startup>1</load-on-startup>
  17. </servlet>
  18. <!--/匹配所有的请求;(不包括.jsp)-->
  19. <!--/*匹配所有的请求;(包括.jsp)-->
  20. <servlet-mapping>
  21. <servlet-name>springmvc</servlet-name>
  22. <url-pattern>/</url-pattern>
  23. </servlet-mapping>
  24. </web-app>
  1. 编写SpringMVC的配置文件!名称:springmvc-servlet.xml:[servletname]-servlet.xml说明,这里的名称要求是按照官方来的 ```xml <?xml version=”1.0” encoding=”UTF-8”?>

  1. 5.
  2. 添加映射器
  3. ```xml
  4. <bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping"/>
  1. 添加处理器适配器
    1. <bean class="org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter"/>
  1. 添加 视图解析器
    1. <!--视图解析器-视图解析器:Dispatcherservlet给他的ModelAndview
    2. 1.获取了ModeLAndView的数据
    3. 2.解析ModeLAndView的视图名字
    4. 3.拼接视图名字,找到对应的视图 /WEB-INF/jsp/hello.jsp
    5. 4.将数据渲染到这个视图
    6. -->
    7. <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" id="InternalResourceViewResolver">
    8. <!--前缀-->
    9. <property name="prefix" value="/WEB-INF/jsp/"/>
    10. <!--后缀-->
    11. <property name="suffix" value=".jsp"/>
    12. </bean>
  1. 编写我们要操作业务Controller,要么实现Controller接口,要么增加注解;需要返回一个ModelAndView,装数据,封视图;

    1. //注意:注意要先导入Controller接口
    2. public class HelloController implements Controller {
    3. @Override
    4. public ModelAndView handleRequest(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) throws Exception {
    5. //ModelAndView 视图与模型
    6. ModelAndView mv = new ModelAndView();
    7. //封装对象,放在ModelAndView中。Model
    8. mv.addObject("msg","HelloSpringMVC");
    9. //封装要跳转的视图,放在ModelAndView中
    10. mv.setViewName("hello"); //:/WEB-INF/jsp/hello.jsp
    11. return mv;
    12. }
    13. }
  1. 将自己的类交给SpringlOC容器,注册bean
    1. <!--Handler-->
    2. <bean id="/hello" class="cn.dafran.controller.HelloController"/>
  1. 写要跳转的jsp页面,显示ModelandView存放的数据,以及我们的正常页面;
    1. <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    2. <html>
    3. <head>
    4. <title>Title</title>
    5. </head>
    6. <body>
    7. ${msg}
    8. </body>
    9. </html>
  1. 配置Tomcat启动测试!

可能遇到的问题:访问出现404,排查步骤:

  1. 查看控制台输出,看一下是不是缺少了什么jar包。

  2. 如果jar包存在,显示无法输出,就在IDEA的项目发布中,添加lib依赖!

  3. 重启Tomcat即可解决!

小结:看这个估计大部分同学都能理解其中的原理了,但是我们实际开发才不会这么写,不然就疯了,还学这个玩意干嘛!我们来看个注解版实现,这才是SpringMVC的精髓,到底有多么简单,看这个图就知道了。

Spring MVC 执行原理

Spring MVC - 图8

  1. 图为SpringMVC的一个较完整的流程图,实线表示SpringMVC框架提供的技术,不需要开发者实现,虚线表示需要开发者实现。

简要分析执行流程

  1. DispatcherServlet表示前置控制器,是整个SpringMVC的控制中心。用户发出请求,
    DispatcherServlet接收请求并拦截请求。

  2. HandlerMapping为处理器映射。DispatcherServlet调用
    HandlerMapping,HandlerMapping根据请求url查找Handler。

  3. HandlerExecution表示具体的Handler,其主要作用是根据url查找控制器,如上url被查找控制器为:hello。

  4. HandlerExecution将解析后的信息传递给DispatcherServlet,如解析控制器映射等。

  5. HandlerAdapter表示处理器适配器,其按照特定的规则去执行Handler。

  6. Handler让具体的Controller执行。

  7. Controller将具体的执行信息返回给HandlerAdapter,如ModelAndView。

  8. HandlerAdapter将视图逻辑名或模型传递给DispatcherServlet。

  9. DispatcherServlet调用视图解析器(ViewResolver)来解析HandlerAdapter传递的逻辑视图名。

  10. 视图解析器将解析的逻辑视图名传给DispatcherServlet。

  11. DispatcherServlet根据视图解析器解析的视图结果,调用具体的视图。

  12. 最终视图呈现给用户。

在这里先听一遍原理,不理解没有关系,我们马上来写一个对应的代码实现大家就明白了,如果不明白,那就写10遍,没有笨人,只有懒人!

注解版

第一步:新建一个Moudle,添加web支持!建立包结构cn.dafran.controller

第二步:由于Maven可能存在资源过滤的问题,我们将配置完善

  1. <build>
  2. <resources>
  3. <resource>
  4. <directory>src/main/java</directory>
  5. <includes>
  6. <include>**/*.properties</include>
  7. <include>**/*.xml</include>
  8. </includes>
  9. <filtering>false</filtering>
  10. </resource>
  11. <resource>
  12. <directory>src/main/resources</directory>
  13. <includes>
  14. <include>**/*.properties</include>
  15. <include>**/*.xml</include>
  16. </includes>
  17. <filtering>false</filtering>
  18. </resource>
  19. </resources>
  20. </build>

第三步:在pom.xml文件引入相关的依赖:

主要有Spring框架核心库、Spring MVC、servlet,JSTL等。我们在父依赖中已经引入了!

第四步:配置web.xml

  1. 注意点:
  • 注意web.xml版本问题,要最新版!

  • 注册DispatcherServlet

  • 关联SpringMVC的配置文件

  • 启动级别为1

  • 映射路径为/【不要用/*,会404】 ```xml <?xml version=”1.0” encoding=”UTF-8”?>

  1. <!--1.注册Dispatcherservlet-->
  2. <servlet>
  3. <servlet-name>SpringMVC</servlet-name>
  4. <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
  5. <!--关联一个springmvc的配置文件:【servlet-name】-servlet.xml-->
  6. <init-param>
  7. <param-name>contextconfigLocation</param-name>
  8. <param-value>classpath:springmvc-servlet.xml</param-value>
  9. </init-param>
  10. <!--启动级别 1 启动顺序,数字越小,启动越早-->
  11. <load-on-startup>1</load-on-startup>
  12. </servlet>
  13. <!--所有请求都会被springmvc拦截-->
  14. <servlet-mapping>
  15. <servlet-name>SpringMVC</servlet-name>
  16. <url-pattern>/</url-pattern>
  17. </servlet-mapping>

  1. > **/和/*的区别:**<br />
  2. /不会匹配到.jsp,只针对我们编写的请求;即:.jsp 不会进入spring Dispatcher Servlet类。<br />
  3. /会匹配 .jsp,<br />
  4. 会出现返回jsp视图时再次进入springDispatcherservlet类,导致找不到对应的controller所以报404错。
  5. > 总之,关于web.xmlurl映射的小知识:<br />
  6. < url-pattern>/ 会匹配到/login这样的路径型url,不会匹配到模式为.jsp这样的后缀型url<br />
  7. < url-pattern>/_ 会匹配所有url:路径型的和后缀型的url(包括/login, _.jsp, _.js _.html等)
  8. 下面是图文版:
  9. ![](https://gitee.com/dafran/pic-go/raw/master/img/image-20210222222438495.png#alt=image-20210222222438495)
  10. ![](https://gitee.com/dafran/pic-go/raw/master/img/image-20210222222647615.png#alt=image-20210222222647615)
  11. ![](https://gitee.com/dafran/pic-go/raw/master/img/image-20210222222058038.png#alt=image-20210222222058038)
  12. **第五步:添加Spring MVC配置文件**
  13. - IOC的注解生效
  14. - 静态资源过滤:HTML.JS.CSS.图片,视频......
  15. - MVC的注解驱动
  16. - 配置视图解析器
  17. resource目录下添加springmvc-servlet.xm配置文件,配置的形式与Spring容器配置基本类似,为了支持基于注解的IOC,设置了自动扫描包的功能,具体配置信息如下:
  18. ```xml
  19. <?xml version="1.0" encoding="UTF-8"?>
  20. <beans xmlns="http://www.springframework.org/schema/beans"
  21. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  22. xmlns:context="http://www.springframework.org/schema/context"
  23. xmlns:mvc="http://www.springframework.org/schema/mvc"
  24. xsi:schemaLocation="http://www.springframework.org/schema/beans
  25. http://www.springframework.org/schema/beans/spring-beans.xsd
  26. http://www.springframework.org/schema/context
  27. https://www.springframework.org/schema/context/spring-context.xsd
  28. http://www.springframework.org/schema/mvc
  29. https://www.springframework.org/schema/mvc/spring-mvc.xsd">
  30. <!--自动扫描包,让指定包下的注解生效,由IOC容器统一管理-->
  31. <context:component-scan base-package="cn.dafran.controller"/>
  32. <!--让Spring MVC不处理静态资源-->
  33. <mvc:default-servlet-handler/>
  34. <!--
  35. 支持mvc注解驱动
  36. 在spring中一般采用@RequestMapping注解来完成映射关系
  37. 要想使@RequestMapping注解生效
  38. 必须向上下文中注册DefaultAnnotationHandlerMapping
  39. 和一个AnnotationMethodHandlerAdapter实例
  40. 这两个实例分别在类级别和方法级别处理。
  41. 而annotation-driven配置帮助我们自动完成上述两个实例的注入。
  42. -->
  43. <mvc:annotation-driven/>
  44. <!--视图解析器-->
  45. <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" id="InternalResourceViewResolver">
  46. <!--前缀-->
  47. <property name="prefix" value="/WEB-INF/jsp/"/>
  48. <!--后缀-->
  49. <property name="suffix" value=".jsp"/>
  50. </bean>
  51. </beans>

在视图解析器中我们把所有的视图都存放在/WEB-INF/目录下,这样可以保证视图安全,因为这个目录下的文件,客户端不能直接访问。

第六步:创建Controller

编写一个Java控制类:cn.dafran.controller.HelloController,注意编码规范

  1. @Controller
  2. @RequestMapping("/HelloController")
  3. public class HelloController {
  4. //真实访问地址:项目名/Hellocontroller/hello
  5. @RequestMapping("/hello")
  6. public String sayHello(Model model){
  7. //向模型中添加属性msg与值,可以在JSP页面中取出并渲染
  8. model.addAttribute("msg","hello,SpringMVC");
  9. //web-inf/jsp/hello.jsp
  10. return "hello";
  11. }
  12. }
  • @Controller是为了让Spring IOC容器初始化时自动扫描到;
  • @RequestMapping是为了映射请求路径,这里因为类与方法上都有映射所以访问时应该是/HelloController/hello;
  • 方法中声明Model类型的参数是为了把Action中的数据带到视图中;
  • 方法返回的结果是视图的名称hello,加上配置文件中的前后缀变成WEB-INF/jsp/hello.jsp。

第七步:创建视图层

在WEB-INF/jsp目录中创建hello.jsp,视图可以直接取出并展示从Controller带回的信息;

可以通过EL表示取出Model中存放的值,或者对象;

  1. <%@ page contentType="text/html;charset=UTF-8" language="java" %>
  2. <html>
  3. <head>
  4. <title>Title</title>
  5. </head>
  6. <body>
  7. ${msg}
  8. </body>
  9. </html>

第八步:配置Tomcat运行

小结
实现步骤其实非常的简单:

  1. 新建一个web项目
  2. 导入相关jar包
  3. 编写web.xml,注册DispatcherServlet
  4. 编写springmvc配置文件
  5. 接下来就是去创建对应的控制类,controner
  6. 最后完善前端视图和controller之间的对应
  7. 测试运行调试.

使用springMVC必须配置的三大件:

处理器映射器处理器适配器视图解析器

通常,我们只需要手动配置视图解析器,而处理器映射器处理器适配器只需要开启注解驱动即可,而省去了大段的xml配置

Controller风格

SpringMVC:Controller及RestFul风格

控制器Controller

  • 控制器复杂提供访问应用程序的行为,通常通过接口定义或注解定义两种方法实现。
  • 控制器负责解析用户的请求并将其转换为一个模型。
  • 在Spring MVC中一个控制器类可以包含多个方法
  • 在Spring MVC中,对于Controller的配置方式有很多种

实现Controller接口

Controller是一个接口,在org.springframework.web.servlet.mvc包下,接口中只有一个方法;

  1. @FunctionalInterface
  2. //实现该接口的类获得控制器功能
  3. public interface Controller {
  4. @Nullable
  5. //处理请求且返回一个模型与视图对象
  6. ModelAndView handleRequest(HttpServletRequest var1, HttpServletResponse var2) throws Exception;
  7. }

测试:

  1. 新建一个Moudle,springmvc-04-controller。将刚才的03拷贝一份,进行操作!
  • 删掉HelloController

  • mvc的配置文件只留下视图解析器!

  1. 编写一个Controller类,ControllerTest1
  1. //只要实现了controller接口的类,说明这就是一个控制器了
  2. public class ControllerTest01 implements Controller {
  3. @Override
  4. public ModelAndView handleRequest(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) throws Exception {
  5. ModelAndView mv = new ModelAndView();
  6. mv.addObject("msg","ControllerTest01");
  7. mv.setViewName("test");
  8. return mv;
  9. }
  10. }
  1. 编写完毕后,去Spring配置文件中注册请求的bean;name对应请求路径,class对应处理请求的类
    1. <bean name="/t1" class="cn.dafran.controller.ControllerTest01"/>
  1. 编写前端test.jsp,注意在WEB-INF/jsp目录下编写,对应我们的视图解析器
  1. <%@ page contentType="text/html;charset=UTF-8" language="java" %>
  2. <html>
  3. <head>
  4. <title>Title</title>
  5. </head>
  6. <body>
  7. ${msg}
  8. </body>
  9. </html>

说明:

  • 实现接口Controller定义控制器是较老的办法
  • 缺点是:一个控制器中只有一个方法,如果要多个方法则需要定义多个Controller;定义的方式比较麻烦;

使用注解 @Controller

  • @Controller注解类型用于声明Spring类的实例是一个控制器(在讲IOC时还提到了另外3个注解);
  • Spring可以使用扫描机制来找到应用程序中所有基于注解的控制器类,为了保证Spring能找到你的控制器,需要在配置文件中声明组件扫描。
  1. <!--自动扫描包,让指定包下的注解生效,由IOC容器统一管理-->
  2. <context:component-scan base-package="cn.dafran.controller"/>
  • 增加一个ControllerTest2类,使用注解实现;
  • 运行测试

    可以发现,我们的两个请求都可以指向一个视图,但是页面结果的结果是不一样的,从这里可以看出视图是被复用的,而控制器与视图之间是弱偶合关系。
    注解方式是平时使用的最多的方式!

RestFul风格

  1. **概念**<br />
  2. Restful就是一个资源定位及资源操作的风格。不是标准也不是协议,只是一种风格。基于这个风格设计的软件可以更简洁,更有层次,更易于实现缓存等机制。
  3. 高效、安全、简洁
  4. **功能**
  1. **使用RESTfu|操作资源:**可以通过不同的请求方式来实现不同的效果!如下:请求地址一样,但是功能可以不同!

学习测试:

  1. 在新建一个类 RestFulController
    1. @Controller
    2. public class RestFulController {
    3. @RequestMapping("/add")
    4. public String test01(int a, int b, Model model) {
    5. int res = a + b;
    6. model.addAttribute("msg","结果为:"+res);
    7. return "test";
    8. }
    9. }
  1. 在Spring MVC中可以使用@PathVariable注解,让方法参数的值对应绑定到一个URI模板变量上。
  1. **使用method属性指定请求类型**<br />

用于约束请求的类型,可以收窄请求范围。指定请求谓词的类型如GET,POST,HEAD,OPTIONS,PUT,PATCH,DELETE,TRACE等

  1. 所有的地址栏请求默认都会是HTTP GET类型的。方法级别的注解变体有如下几个:组合注解@GetMapping是一个组合注解<br />

它所扮演的是@RequestMapping(method=RequestMethod.GET)的一个快捷方式。
平时使用的会比较多!

  1. @GetMapping
  2. @PostMapping
  3. @PutMapping
  4. @DeleteMapping
  5. @PatchMapping
  6. //@RequestMapping(value="/add/{a}/{b}",method=RequestMethod.GET);
  7. //value或者path

小黄鸭调试法

  1. 场景一:我们都有过向别人(甚至可能向完全不会编程的人)提问及解释编程问题的经历,但是很多时候就在我们解释的过程中自己却想到了问题的解决方案,然后对方却一脸茫然。
  2. 场景二:你的同行跑来问你一个问题,但是当他自己把问题说完,或说到一半的时候就想出答案走了,留下一脸茫然的你。
  3. 其实上面两种场景现象就是所谓的小黄鸭调试法(Rubber Duck Debuging),又称橡皮鸭调试法,它是我们软件工程中最常使用调试方法之一。

SpringMVC:结果跳转方式

ModelAndview

设置 Modelandview对象,根据vew的名称,和视图解析器跳到定的页面

页面:{视图解析器前缀}+ viewname+{视图解析器后缀}

  1. <!--视图解析器-->
  2. <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" id="InternalResourceViewResolver">
  3. <!--前缀-->
  4. <property name="prefix" value="/WEB-INF/jsp/"/>
  5. <!--后缀-->
  6. <property name="suffix" value=".jsp"/>
  7. </bean>

对应的controller类

  1. //只要实现了controller接口的类,说明这就是一个控制器了
  2. public class ControllerTest01 implements Controller {
  3. @Override
  4. public ModelAndView handleRequest(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) throws Exception {
  5. ModelAndView mv = new ModelAndView();
  6. mv.addObject("msg","ControllerTest01");
  7. mv.setViewName("test");
  8. return mv;
  9. }
  10. }

ServletAPl

  1. 通过设置 ServletAPI,不需要视图解析器
  1. 通过HttpServletResponse进行输出
  2. 通过HttpServletResponse实现重定向
  3. 通过HttpServletResponse实现转发
  1. @controller
  2. public class ResultGo {
  3. @RequestMapping("/result/t1")
  4. public void test1(HttpServletRequest req, HttpServletResponse rsp) throws IOException {
  5. rsp.getwriter().println("Hello,Spring by servlet API");
  6. }
  7. @RequestMapping("/result/t2")
  8. public void test2(HttpServletRequest req, HttpServletResponse rsp) throws IOException {
  9. rsp.sendRedirect("/index.jsp");
  10. }
  11. @RequestMapping("/result/t3")
  12. public void test3(HttpServletRequest req, HttpServletResponse rsp) throws Exception {
  13. //转发
  14. req.setattribute("msg","/result/t3");
  15. req.getRequestDispatcher("/WEB-INF/jsp/test.jsp").forward(req,rsp);
  16. }
  17. }

Servlet(req,resp) (请求,响应)

Spring MVC | doService(req,resp)

Spring MVC

  1. **通过SpringMVC来实现转发和重定向-无需视图解析器;**
  2. 测试前,需要将视图解析器注释掉
  1. @Controller
  2. public class RestFulController {
  3. @RequestMapping("/rsm/t1")
  4. public String test1(Model model) {
  5. model.addAttribute("msg","msg");
  6. //转发
  7. return "/index.jsp";
  8. }
  9. @RequestMapping("/rsm/t2")
  10. public String test2() {
  11. //转发二
  12. return "forward:/index.jsp";
  13. }
  14. @RequestMapping("/rsm/t3")
  15. public String test3() {
  16. //重定向
  17. return "redirect:/index.jsp";
  18. }
  19. }
  1. **通过SpringMVC来实现转发和重定向-有视图解析器;**<br />

重定向,不需要视图解析器,本质就是重新请求一个新地方,所以注意路径问题.可以重定向到另外一个请求实现.

  1. @Controller
  2. public class ResultspringMVC2 {
  3. @RequestMapping("/rsm2/t1")
  4. public string test1(){
  5. //转发
  6. return"test";
  7. }
  8. @RequestMapping("/rsm2/t2")
  9. public string test2(){
  10. //重定向
  11. return"redirect:/index.jsp";
  12. //return"redirect:hello.do";//hello.do为另一个请求/
  13. }
  14. }

SpringMVC:数据处理

处理提交数据

  1. **1、提交的域名称和处理方法的参数名一致**
  2. 提交数据:[http://localhost:8080/hello?name=dafran](http://localhost:8080/hello?name=dafran)
  3. 处理方法:
  1. @RequestMapping("/hello")
  2. public String hello(String name) {
  3. System.out.println(name);
  4. return "hello";
  5. }

后台输出:dafran

  1. **2、提交的域名称和处理方法的参数名不一致**
  2. 提交数据:[http://localhost:8080/hello?username=dafran](http://localhost:8080/hello?username=dafran)
  3. 处理方法:
  1. //@RequestParam("username") : username提交的域的名称
  2. @RequestMapping("/hello")
  3. public String hello(@RequestParam("username") String name) {
  4. System.out.println(name);
  5. return "hello";
  6. }

后台输出:dafran

  1. **3、提交的是一个对象**<br />
  2. 要求提交的表单域和对象的属性名一致,参数使用对象即可

1.实体类

  1. public class User {
  2. private int id;
  3. private String name;
  4. private int age;
  5. //构造
  6. //get/set
  7. //tostring()
  8. }

2.提交数据:http://localhost:8080/mvc04/user?name=dafran&id=1&age=15

3.处理方法:

  1. /*
  2. 1.接收前端用户传递的参数,判断参数的名字,假设名字直接在方法上,可以直接使用
  3. 2.假设传递的是一个对象User,匹配User对象中的字段名;如果名字一致则OK,否则,匹配不到
  4. */
  5. @RequestMapping("/user")
  6. public String user(User user) {
  7. System.out.println(user);
  8. return "hello";
  9. }

后台输出:User{id=1,name=’dafran’,age=15}

说明:如果使用对象的话,前端传递的参数名和对象名必须一致,否则就是null。

LinkedHashMap

ModelMap:继承了LinkedHashMap,所以他拥有LinkedHashMap的全部功能!

Model:精简版(大部分情况,我们都直接使用Model)

对比

  1. 就对于新手而言简单来说使用区别就是:
  1. Model只有塞塞几个方法只适合用于储存数据,简化了新手对于Model对象的操作和理解;
  2. ModelMap继承了LinkedMap,除了实现了自身的一些方法,同样的继承LinkedMap的方法和特性;
  3. ModelAndview可以在储存数据的同时,可以进行设置返回的逻辑视图,进行控制展示层的跳转。
  1. 当然更多的以后开发考虑的更多的是性能和优化,就不能单单仅限于此的了解。
  2. **请使用80%的时间打好扎实的基础,剩下18%的时间研究框架,2%的时间去学点英文,框架的官方文档永远是最好的教程。**

乱码处理

测试步骤:

  1. 我们可以在首页编写一个提交的表单
    1. <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    2. <html>
    3. <head>
    4. <title>Title</title>
    5. </head>
    6. <body>
    7. <form action="/e/t1" method="post">
    8. <input type="text" name="name">
    9. <input type="submit">
    10. </form>
    11. </body>
    12. </html>
  1. 后台编写对应的处理类

    1. @Controller
    2. public class EncodingController {
    3. @PostMapping("/e/t1")
    4. public String test01(String name,Model model) {
    5. model.addAttribute("msg",name); //获取表单提交的值
    6. return "test"; //跳转到test页面显示输入的值
    7. }
    8. }
  1. 输入中文测试,发现乱码
  1. 不得不说,乱码问题是在我们开发中十分常见的问题,也是让我们程序猿比较头大的问题!以前乱码问题通过过滤器解决,而SpringMVC给我们提供了一个过滤器,可以在web.xml中配置.修改了xml文件需要重启服务器!
  1. <!--过滤器-->
  2. <filter>
  3. <filter-name>encoding</filter-name>
  4. <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
  5. </filter>
  6. <filter-mapping>
  7. <filter-name>encoding</filter-name>
  8. <url-pattern>/*</url-pattern>
  9. </filter-mapping>
  1. 有些极端情况下.这个过滤器对get的支持不好.

处理方法:

1、修改tomcat配置文件 :设置编码!

  1. <Connector URIEncoding="utf-8" port="8080" protocol="HTTP/1.1"
  2. connectionTimeout="20000"
  3. redirectPort="8443"/>

2、自定义过滤器

  1. package com.kuang.filter;
  2. import javax.servlet.*;
  3. import javax.servlet.http.HttpServletRequest;
  4. import javax.servlet.http.HttpServletRequestWrapper;
  5. import javax.servlet.http.HttpServletResponse;
  6. import java.io.IOException;
  7. import java.io.UnsupportedEncodingException;
  8. import java.util.Map;
  9. /**
  10. * 解决get和post请求 全部乱码的过滤器
  11. */
  12. public class GenericEncodingFilter implements Filter {
  13. @Override
  14. public void destroy() {
  15. }
  16. @Override
  17. public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
  18. //处理response的字符编码
  19. HttpServletResponse myResponse=(HttpServletResponse) response;
  20. myResponse.setContentType("text/html;charset=UTF-8");
  21. // 转型为与协议相关对象
  22. HttpServletRequest httpServletRequest = (HttpServletRequest) request;
  23. // 对request包装增强
  24. HttpServletRequest myrequest = new MyRequest(httpServletRequest);
  25. chain.doFilter(myrequest, response);
  26. }
  27. @Override
  28. public void init(FilterConfig filterConfig) throws ServletException {
  29. }
  30. }
  31. //自定义request对象,HttpServletRequest的包装类
  32. class MyRequest extends HttpServletRequestWrapper {
  33. private HttpServletRequest request;
  34. //是否编码的标记
  35. private boolean hasEncode;
  36. //定义一个可以传入HttpServletRequest对象的构造函数,以便对其进行装饰
  37. public MyRequest(HttpServletRequest request) {
  38. super(request);// super必须写
  39. this.request = request;
  40. }
  41. // 对需要增强方法 进行覆盖
  42. @Override
  43. public Map getParameterMap() {
  44. // 先获得请求方式
  45. String method = request.getMethod();
  46. if (method.equalsIgnoreCase("post")) {
  47. // post请求
  48. try {
  49. // 处理post乱码
  50. request.setCharacterEncoding("utf-8");
  51. return request.getParameterMap();
  52. } catch (UnsupportedEncodingException e) {
  53. e.printStackTrace();
  54. }
  55. } else if (method.equalsIgnoreCase("get")) {
  56. // get请求
  57. Map<String, String[]> parameterMap = request.getParameterMap();
  58. if (!hasEncode) { // 确保get手动编码逻辑只运行一次
  59. for (String parameterName : parameterMap.keySet()) {
  60. String[] values = parameterMap.get(parameterName);
  61. if (values != null) {
  62. for (int i = 0; i < values.length; i++) {
  63. try {
  64. // 处理get乱码
  65. values[i] = new String(values[i]
  66. .getBytes("ISO-8859-1"), "utf-8");
  67. } catch (UnsupportedEncodingException e) {
  68. e.printStackTrace();
  69. }
  70. }
  71. }
  72. }
  73. hasEncode = true;
  74. }
  75. return parameterMap;
  76. }
  77. return super.getParameterMap();
  78. }
  79. //取一个值
  80. @Override
  81. public String getParameter(String name) {
  82. Map<String, String[]> parameterMap = getParameterMap();
  83. String[] values = parameterMap.get(name);
  84. if (values == null) {
  85. return null;
  86. }
  87. return values[0]; // 取回参数的第一个值
  88. }
  89. //取所有值
  90. @Override
  91. public String[] getParameterValues(String name) {
  92. Map<String, String[]> parameterMap = getParameterMap();
  93. String[] values = parameterMap.get(name);
  94. return values;
  95. }
  96. }

这个也是我在网上找的一些大神写的,一般情况下,SpringMVC默认的乱码处理就已经能够很好的解决了!

然后在web.xml中配置这个过滤器即可!

乱码问题,需要平时多注意,在尽可能能设置编码的地方,都设置为统一编码 UTF-8!

有了这些知识,我们马上就可以进行SSM整合了!

Json

什么是JSON?

  • JSON(JavaScript Object Notation,JS对象标记)是一种轻量级的数据交换格式,目前使用特别广泛。
  • 采用完全独立于编程语言的文本格式来存储和表示数据。
  • 简洁和清晰的层次结构使得JSON成为理想的数据交换语言。
  • 易于人阅读和编写,同时也易于机器解析和生成,并有效地提升网络传输效率。

    在JavaScript语言中,一切都是对象。因此,任何JavaScript支持的类型都可以通过JSON来表示,例如字符串、数字、对象、数组等。看看他的要求和语法格式:

  • 对象表示为键值对,数据由逗号分隔

  • 花括号保存对象
  • 方括号保存数组

    JSON键值对是用来保存JavaScript对象的一种方式,和JavaScript对象的写法也大同小异,键/值对组合中的键名写在前面并用双引号””包裹,使用冒号:分隔,然后紧接着值:

前后端分离时代:

后端部署后端,提供接口,提供数据:

  1. json

前端独立部署,负责渲染后端的数据:

  1. {"name":"6b92d6"}
  2. {"age":"3"}
  3. {"sex":"男"}
  1. 很多人搞不清楚JSONJavaScript对象的关系,甚至连谁是谁都不清楚。其实,可以这么理解:
  • JSON是JavaScript对象的字符串表示法,它使用文本表示一个JS对象的信息,本质是一个字符串。
  1. var obj={a:'Hello',b:'world'}; //这是一个对象,注意键名也是可以使用引号包裹的
  2. var json='{"a":"Hello","b":"World"}'; //这是一个JSON字符串,本质是一个字符串
  1. **JSONJavaScript对象互转**
  • 要实现从JSON字符串转换为JavaScript对象,使用JSON.parse()方法:
    1. var obj = JSON.parse('{"a":"Hello","b":"world"}');
    2. //结果是{a:'Hello',b:'World'}
  • 要实现从JavaScript对象转换为JSON字符串,使用JSON.stringify()方法:
    1. var json=JSON.stringify({a:'Hello',b:'World'});
    2. //结果是'{"a":"Hello","b":"World"}'

Controller返回JSON数据

  • Jackson应该是目前比较好的json解析工具了

  • 当然工具不止这一个,比如还有阿里巴巴的fastjson等等。

  • 我们这里使用Jackson,使用它需要导入它的jar包; ```xml

    2.12.1 1.16.18

com.fasterxml.jackson.core jackson-databind ${jackson-databind.version} org.projectlombok lombok ${lombok.version}

  1. -
  2. 配置SpringMVC需要的配置
  3. <br />**web.xml**
  4. ```xml
  5. <?xml version="1.0" encoding="UTF-8"?>
  6. <web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
  7. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  8. xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
  9. version="4.0">
  10. <!--配置DispatchServlet-->
  11. <servlet>
  12. <servlet-name>springmvc</servlet-name>
  13. <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
  14. <init-param>
  15. <param-name>contextConfigLocation</param-name>
  16. <param-value>classpath:spring-servlet.xml</param-value>
  17. </init-param>
  18. <load-on-startup>1</load-on-startup>
  19. </servlet>
  20. <servlet-mapping>
  21. <servlet-name>springmvc</servlet-name>
  22. <url-pattern>/</url-pattern>
  23. </servlet-mapping>
  24. <!--过滤器-->
  25. <filter>
  26. <filter-name>encoding</filter-name>
  27. <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
  28. </filter>
  29. <filter-mapping>
  30. <filter-name>encoding</filter-name>
  31. <url-pattern>/</url-pattern>
  32. </filter-mapping>
  33. </web-app>

spring-servlet.xml

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xmlns:context="http://www.springframework.org/schema/context"
  5. xmlns:mvc="http://www.springframework.org/schema/mvc"
  6. xsi:schemaLocation="http://www.springframework.org/schema/beans
  7. http://www.springframework.org/schema/beans/spring-beans.xsd
  8. http://www.springframework.org/schema/context
  9. https://www.springframework.org/schema/context/spring-context.xsd
  10. http://www.springframework.org/schema/mvc
  11. https://www.springframework.org/schema/mvc/spring-mvc.xsd">
  12. <!--自动扫描包,让指定包下的注解生效,由IOC容器统一管理-->
  13. <context:component-scan base-package="cn.dafran.controller"/>
  14. <mvc:default-servlet-handler/>
  15. <mvc:annotation-driven/>
  16. <!--视图解析器-->
  17. <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" id="InternalResourceViewResolver">
  18. <!--前缀-->
  19. <property name="prefix" value="/WEB-INF/jsp/"/>
  20. <!--后缀-->
  21. <property name="suffix" value=".jsp"/>
  22. </bean>
  23. </beans>

编写一个实体类

  1. @Data
  2. @AllArgsConstructor
  3. @NoArgsConstructor
  4. public class User {
  5. private String name;
  6. private int age;
  7. private String sex;
  8. }

编写一个Controller

  1. @Controller
  2. public class UserController {
  3. @RequestMapping("/json1")
  4. @ResponseBody //它就不会走视图解析器,会直接返回一个字符串
  5. public String json1() throws JsonProcessingException {
  6. //jackson,ObjectMapper
  7. //创建一个javason的对象映射器,用来解析数据。
  8. ObjectMapper mapper = new ObjectMapper();
  9. //创建一个对象
  10. User user = new User("001", 1, "男");
  11. //将对象解析成Json格式
  12. String str = mapper.writeValueAsString(user);
  13. //由于@ResponesBody注解,会将str转换成json格式返回
  14. return str;
  15. }
  16. }

出现乱码

  1. //produces:指定响应体返回类型和编码
  2. @RequestMapping(value = "/json1", produces = "application/json;charset=utf-8")

【注意:使用json记得处理乱码问题】

代码优化

  1. 乱码统一优化
  2. 上一种方法比较麻烦,如果项目中有许多请求则每一个都要添加,可以通过Spring配置统一指定,这样就不用每次都去处理了!
  3. 我们可以在springmvc的配置文件上添加一段消息StringHttpMessageConverter转换配置!
  1. <!--JSON乱码问题配置-->
  2. <mvc:annotation-driven>
  3. <mvc:message-converters register-defaults="true">
  4. <bean class="org.springframework.http.converter.StringHttpMessageConverter">
  5. <constructor-arg value="UTF-8"/>
  6. </bean>
  7. <bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter">
  8. <property name="objectMapper">
  9. <bean class="org.springframework.http.converter.json.Jackson2ObjectMapperFactoryBean">
  10. <property name="failOnEmptyBeans" value="false"/>
  11. </bean>
  12. </property>
  13. </bean>
  14. </mvc:message-converters>
  15. </mvc:annotation-driven>

返回Json字符串统一解决

  1. 类上直接加`@RestController` 这样里面的所有方法都只返回Json字符串不用单独的添加`@ResponseBody`
  1. @RestController
  2. public class UserController {
  3. @RequestMapping("/j1")
  4. public String json1() throws JsonProcessingException {
  5. //jackson,ObjectMapper
  6. //创建一个javason的对象映射器,用来解析数据。
  7. ObjectMapper mapper = new ObjectMapper();
  8. //创建一个对象
  9. User user = new User("001", 1, "男");
  10. //将对象解析成Json格式
  11. String str = mapper.writeValueAsString(user);
  12. //由于@ResponesBody注解,会将str转换成json格式返回
  13. return str;
  14. }
  15. }

测试集合输出

  1. @RequestMapping("/json2")
  2. public String json2() throws JsonProcessingException {
  3. //jackson,ObjectMapper
  4. //创建一个javason的对象映射器,用来解析数据。
  5. ObjectMapper mapper = new ObjectMapper();
  6. ArrayList<User> users = new ArrayList<>();
  7. User user1 = new User("001", 1, "男");
  8. User user2 = new User("002", 1, "男");
  9. User user3 = new User("003", 1, "男");
  10. User user4 = new User("004", 1, "男");
  11. users.add(user1);
  12. users.add(user2);
  13. users.add(user3);
  14. users.add(user4);
  15. //将对象解析成Json格式
  16. String str = mapper.writeValueAsString(users);
  17. //由于@ResponesBody注解,会将str转换成json格式返回
  18. return str;
  19. }

输出时间对象

  1. @RequestMapping("/j3")
  2. public String json3() throws JsonProcessingException {
  3. ObjectMapper mapper = new ObjectMapper();
  4. Date date = new Date();
  5. //自定义日期时间格式
  6. SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
  7. //ObjectMapper,时间解析后的默认格式为:Timestamp,时间戳
  8. return mapper.writeValueAsString(sdf.format(date));
  9. }

运行结果:

  • 默认日期格式会变成一个数字,是1970年1月1日到当前日期的毫秒数!
  • Jackson默认是会把时间转成timestamps形式

    解决方案:取消timestamps形式,自定义时间格式

  1. @RequestMapping("/j4")
  2. public String json4() throws JsonProcessingException {
  3. ObjectMapper mapper = new ObjectMapper();
  4. //不使用时间戳的方式
  5. mapper.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS,false);
  6. //自定义日期时间格式
  7. SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
  8. //指定日期格式
  9. mapper.setDateFormat(sdf);
  10. Date date = new Date();
  11. return mapper.writeValueAsString(date);
  12. }

工具类

实现代码复用,提高代码精简度,在重写的方法给默认值

  1. public class JsonUtils {
  2. public static String getJson(Object object) {
  3. return getJson(object,"yyyy-MM-dd HH:mm:ss");
  4. }
  5. public static String getJson(Object object, String dateFormat) {
  6. ObjectMapper mapper = new ObjectMapper();
  7. //不使用时间戳的方式
  8. mapper.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS,false);
  9. //自定义日期时间格式
  10. SimpleDateFormat sdf = new SimpleDateFormat(dateFormat);
  11. //指定日期格式
  12. mapper.setDateFormat(sdf);
  13. try {
  14. return mapper.writeValueAsString(object);
  15. } catch (JsonProcessingException e) {
  16. e.printStackTrace();
  17. }
  18. return null;
  19. }
  20. }

FastJson

fastjson.jar是阿里开发的一款专门用于Java开发的包,可以方便的实现json对象与JavaBean对象的转换,实现JavaBean对象与json字符串的转换,实现json对象与json字符串的转换。实现json的转换方法很多,最后的实现结果都是一样的。

导入依赖

  1. <!--FastJson-->
  2. <dependency>
  3. <groupId>com.alibaba</groupId>
  4. <artifactId>fastjson</artifactId>
  5. <version>1.2.60</version>
  6. </dependency>

fastjson 三个主要的类:

  • 【NSONObject代表json对象】

    • JSONObject实现了Map接口,猜想JSONObject底层操作是由Map实现的。
    • JSONObject对应json对象,通过各种形式的get()方法可以获取json对象中的数据,也可利用诸如size(),isEmpty()等方法获取“键:值”对的个数和判断是否为空。其本质是通过实现Map接口并调用接口中的方法完成的。
  • 【JSONArray代表json对象数组】

    • 内部是有List接口中的方法来完成操作的。
  • 【JSON代表JSONObject和JSONArray的转化】

    • JSON类源码分析与使用
    • 仔细观察这些方法,主要是实现json对象,json对象数组,javabean对象,json字符串之间的相互转化。
  1. System.out.println("*******Java对象转JSON字符串********);
  2. String str1 = JSON. toJSONString(list);
  3. System.out.println("JSON.toJSONString(list)==>"+str1);
  4. String str2 = JSON. toJSONString(user1)
  5. System.out.println("JSON.toJSONString(user1)==>"+str2);
  6. System.out.println("\n*******JSON字符串转Java对象********);
  7. User jp_user1=JSON.parseObject(str2,User.class);
  8. System.out.println("JSON.parseObject(str2,User.class)==>"+jp_user1);
  9. System.out.println("\n*******Java对象转JSON对象******");
  10. JSONObject jsonobject1 = (JSONObject) JSON. toJSON(user2);
  11. System.out.println("(JSONObject)JSON.toJSON(user2)==>"+jsonobject1.getstring(key:"name");
  12. System.out.println("\n******JSON对象转Java对象******");
  13. User to_java_user=JSON.toJavaobject(jsonobject1,User.class);
  14. System.out.println("JSON.toJavaobject(jsonobject1,User.class)==>"+to_java_user);

SpringMVC:拦截器

概述

  1. SpringMVC的处理器拦截器类似于Servlet开发中的过滤器Filter,用于对处理器进行预处理和后处理。开发者可以自己定义一些拦截器来实现特定的功能。
  2. **过滤器与拦截器的区别**:拦截器是AOP思想的具体应用。
  3. **过滤器**
  • servlet规范中的一部分,任何java web工程都可以使用

  • 在url-pattern中配置了/之后,可以对所有要访问的资源进行拦截
    *拦截器

  • 拦截器是SpringMVC框架自己的,只有使用了SpringMVC框架的工程才能使用

  • 拦截器只会拦截访问的控制器方法,如果访问的是jsp/html/css/image/js是不会进行拦截的

自定义拦截器

  1. 那如何实现拦截器呢?
  2. 想要自定义拦截器,必须实现`Handlerlntercetor`接口。
  1. 新建一个Moudule,springmvc-07-Interceptor,添加web支持

  2. 配置web.xml和springmvc-servlet.xml文件

  3. 编写一个拦截器

    1. public class MyInterceptor implements HandlerInterceptor {
    2. //return ture;执行下一个拦截器,放行
    3. //return fales;不执行下一个拦截器
    4. @Override
    5. public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
    6. System.out.println("=====处理前=====");
    7. return false;
    8. }
    9. //日志
    10. @Override
    11. public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
    12. System.out.println("=======处理后=========");
    13. }
    14. @Override
    15. public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
    16. System.out.println("========清理========");
    17. }
    18. }

文件上传与下载

文件上传

commons-fileupload

一、导入文件上传的jar包,commons-fileupload,Maven会自动帮我们导入他的依赖包commons-io包;

  1. <!--文件上传-->
  2. <dependency>
  3. <groupId>commons-fileupload</groupId>
  4. <artifactId>commons-fileupload</artifactId>
  5. <version>1.3.3</version>
  6. </dependency>
  7. <!--servlet-api导入高版本-->
  8. <dependency>
  9. <groupId>javax.servlet</groupId>
  10. <artifactId>javax.servlet-api</artifactId>
  11. <version>4.0.1</version>
  12. </dependency>

二、配置bean:multipartResolver

  1. 【注意!!!这个benaid必须为:multipartResolver,否则上传文件会报400的错误!在这里栽过坑,教训!】
  1. <!--文件上传配置-->
  2. <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
  3. <!--请求的编码格式,必须和jSP的pageEncoding属性一致,以便正确读取表单的内容,默认为ISO-8859-->
  4. <property name="defaultEncoding" value="utf-8"/>
  5. <!--上传文件大小上限,单位为字节(10485760=10M)-->
  6. <property name="maxUploadSize" value="10485760"/>
  7. <property name="maxInMemorySize" value="40960"/>
  8. </bean>
  1. CommonsMultipartFile的常用方法:
  • String getOriginalFilename():获取上传文件的原名
  • InputStream getlnputStream():获取文件流
  • void transferTo(File dest):将上传文件保存到一个目录文件中

三、编写前端页面

  1. <form action="/upload"enctype="multipart/form-data"method="post">
  2. <input type="file" name="file"/>
  3. <input type="submit" value="upload">
  4. </form>

四、Controller

  1. @RestController
  2. public class FileController {
  3. //@RequestParam("file")将name=file控件得到的文件封装成CommonsMultipartFile对象
  4. //批量上传CommonsMultipartFile则为数组即可
  5. @RequestMapping("/upload")
  6. public String fileUpload(@RequestParam("file") CommonsMultipartFile file, HttpServletRequest request) throws IOException {
  7. //获取文件名:file.getoriginalFilename();
  8. String uploadFileName = file.getOriginalFilename();
  9. //如果文件名为空,直接回到首页!
  10. if ("".equals(uploadFileName)) {
  11. return "redirect:/index.jsp";
  12. }
  13. System.out.println("文件上传名: " + uploadFileName);
  14. //上传路径保存设置 UUID
  15. String path = request.getServletContext().getRealPath("/upload");
  16. //如果路径不存在,创建一个
  17. File realPath = new File(path);
  18. if (!realPath.exists()){
  19. realPath.mkdirs();
  20. }
  21. System.out.println("上传文件保存地址: " + realPath);
  22. InputStream is = file.getInputStream(); //文件输入流
  23. OutputStream os = new FileOutputStream(new File(realPath, uploadFileName)); //文件输出流
  24. //读取写出
  25. int len=0;
  26. byte[] buffer = new byte[1024];
  27. while ((len= is.read(buffer)) != -1) {
  28. os.write(buffer,0,len);
  29. os.flush();
  30. }
  31. os.close();
  32. is.close();
  33. return "redirect:/index.jsp";
  34. }
  35. }

五、测试上传文件。

采用file.Transto来保存上传的文件

  1. 编写Controller

    1. public String fileUploaded2(CommonsMultipartFile file, HttpServletRequest request) throws IOException {
    2. //上传保存路径
    3. String path = request.getServletContext().getRealPath("/upload");
    4. File realPath = new File(path);
    5. if (!realPath.exists()) {
    6. realPath.mkdirs();
    7. }
    8. System.out.println("上传文件保存地址: " + realPath);
    9. //通过CommonsMultipartFile的方法直接写文件(注意这个时候)
    10. file.transferTo(new File(realPath+"/"+file.getOriginalFilename()));
    11. return "redirect:/index.jsp";
    12. }
  1. 前端表单提交地址修改

  2. 访问提交测试,OK!

文件下载

文件下载步骤:

1、设置 response 响应头

2、读取文件 — InputStream

3、写出文件 — OutputStream

4、执行操作

5、关闭流 (先开后关)

代码实现:

  1. @RequestMapping(value="/download")
  2. public String downloads(HttpServletResponse response ,HttpServletRequest request) throws Exception{
  3. //要下载的图片地址
  4. String path = request.getServletContext().getRealPath("/upload");
  5. String fileName = "基础语法.jpg";
  6. //1、设置response 响应头
  7. response.reset(); //设置页面不缓存,清空buffer
  8. response.setCharacterEncoding("UTF-8"); //字符编码
  9. response.setContentType("multipart/form-data"); //二进制传输数据
  10. //设置响应头
  11. response.setHeader("Content-Disposition",
  12. "attachment;fileName="+URLEncoder.encode(fileName, "UTF-8"));
  13. File file = new File(path,fileName);
  14. //2、 读取文件--输入流
  15. InputStream input=new FileInputStream(file);
  16. //3、 写出文件--输出流
  17. OutputStream out = response.getOutputStream();
  18. byte[] buff =new byte[1024];
  19. int index=0;
  20. //4、执行 写出操作
  21. while((index= input.read(buff))!= -1){
  22. out.write(buff, 0, index);
  23. out.flush();
  24. }
  25. out.close();
  26. input.close();
  27. return null;
  28. }

前端

  1. <a href="/download">点击下载</a>

测试,文件下载OK。