ssm-springmvc-学习笔记

简介

简单的来说,就是一个在表述层负责和前端数据进行交互的框架

帮我们简化了许多从前端获取数据的步骤

springmvc基本流程

用户在原本的没有框架的时候请求会直接调用到controller这个类,但是其步骤非常繁琐

所以我们就使用springmvc进行简化

当用户发送请求时,首先到达的是Servlet,解析用户的请求,然后去mapping类寻找用户请求的是类中的哪一个方法

接着寻找到是哪个类之后,在发送request到Adapter去,此时adapter会拆解request然后将其中有用的参数发送给handler执行对应的逻辑

接着handler执行完毕后再发送adapter处理JSON串给servlet一个respone报文,返回给前端做出数据响应

核心组件

  1. DispatcherServlet : SpringMVC提供,我们需要使用web.xml配置使其生效,它是整个流程处理的核心,所有请求都经过它的处理和分发![ CEO ]

  2. HandlerMapping : SpringMVC提供,我们需要进行IoC配置使其加入IoC容器方可生效,它内部缓存handler(controller方法)和handler访问路径数据,被DispatcherServlet调用,用于查找路径对应的handler![秘书]

  3. HandlerAdapter : SpringMVC提供,我们需要进行IoC配置使其加入IoC容器方可生效,它可以处理请求参数和处理响应数据数据,每次DispatcherServlet都是通过handlerAdapter间接调用handler,他是handler和DispatcherServlet之间的适配器![经理]

  4. Handler : handler又称处理器,他是Controller类内部的方法简称,是由我们自己定义,用来接收参数,向后调用业务,最终返回响应结果![打工人]

  5. ViewResovler : SpringMVC提供,我们需要进行IoC配置使其加入IoC容器方可生效!视图解析器主要作用简化模版视图页面查找的,但是需要注意,前后端分离项目,后端只返回JSON数据,不返回页面,那就不需要视图解析器!所以,视图解析器,相对其他的组件不是必须的![财务]

springmvc运用大致流程

首先声明一下controller

通过requestMapping指定该类的访问路径

java 复制代码
package com.atguigu.controller;


import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;

@Controller
public class HelloController {

//     对外访问的地址,到handlerMapping注册的注册
    @RequestMapping("springmvc/hello")
    @ResponseBody//直接返回字符串给前端不需要找视图
    public String Hello()
    {
        System.out.println("hello");
        return "springMvc Hello";
    }
}

接着我们声明一个配置类,将各种springmvc需要的类放入ioc容器当中

通过Bean组件放入ioc容器

java 复制代码
//TODO: 导入handlerMapping和handlerAdapter的三种方式
//1.自动导入handlerMapping和handlerAdapter [推荐]
//2.可以不添加,springmvc会检查是否配置handlerMapping和handlerAdapter,没有配置默认加载
//3.使用@Bean方式配置handlerMapper和handlerAdapter
@EnableWebMvc
@Configuration
@ComponentScan(basePackages = "com.atguigu.controller") //TODO: 进行controller扫
//WebMvcConfigurer springMvc进行组件配置的规范,配置组件,提供各种方法! 前期可以实现
public class MvcConfig implements WebMvcConfigurer {

    @Bean
    public HandlerMapping handlerMapping(){
        return new RequestMappingHandlerMapping();
    }

    @Bean
    public HandlerAdapter handlerAdapter(){
        return new RequestMappingHandlerAdapter();
    }

}

那么我们知道spring中将类放入ioc容器中是需要读取一些配置文件的

那么将springmvc需要的各种工具类放入ioc容器中依靠的是如下的接口

复制代码
AbstractAnnotationConfigDispatcherServletInitializer
java 复制代码
package com.atguigu.config;

import org.springframework.web.servlet.support.AbstractAnnotationConfigDispatcherServletInitializer;

//该类可以被web项目加载,会初始化ioc容器,会设置dispatcherServlet的地址
public class SpringMvcInit extends AbstractAnnotationConfigDispatcherServletInitializer {
    @Override
    protected Class<?>[] getRootConfigClasses() {
        return new Class[0];
    }


//    设置我们项目的配置类
    @Override
    protected Class<?>[] getServletConfigClasses() {
        return new Class[]{MvcConfig.class};
    }


//    配置springmvc内部自带的servlet的访问地址
    @Override
    protected String[] getServletMappings() {
        return new String[]{"/"};
//        单个斜杠就代表着处理所有的用户请求的意思
    }
}

那么上述的方法是如何实现将类放入ioc容器中呢?

我们先介绍一个接口

这个接口里面有一个onStartUp函数,其中这个函数会在web项目启动的时候调用

web容器初始化讲解

我们来找寻

复制代码
AbstractAnnotationConfigDispatcherServletInitializer

的继承关系然后找出web容器初始化的原理

当我们点进去它的父类之后,我们再进去它的父类观察

再父类里面我们发现有一个

复制代码
registerDispatcherServlet

方法,这个方法就是将web容器放入ioc容器当中的方法,我们点进去观察

点进去这个方法之后,我们发现有一个名为

复制代码
createServletApplicationContext();

的方法,我们再点进去观察

我们在这里发现,有一个

复制代码
AnnotationConfigWebApplicationContext();

类,这个类再spring中是用于读取配置类和配置文件将类导入ioc容器中的类

紧接着我们发现,在这个类中,我们发现了一个

复制代码
getServletConfigClasses();

这个方法。是不是很熟悉?

我们回到最开始的地方观察

在最开始的继承了上述所有类的子类中,是不是有一个这样的方法

将配置类通过反射装入数组,然后再父类的函数中调用数组将其放入了ioc容器

如果对象数组不为空就将其配置类放入ioc容器进行注册

里面还有servletmapping的代码,用于将地址映射的类也装入ioc容器中

SpringMvc接收数据

1.访问路径设置

Java代码展示

该代码中和新注释就是@ResquestMapping

该注释能够使得这个controller响应一个user的网页路径请求

并且写在该类下的所有方法的请求路径都是

/user/方法上的request路径

其余的原理都写在了注释里

java 复制代码
package com.atguigu.requestmapping;


import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;

@Controller
@RequestMapping("user")
public class UserController {

//    handler->handlerMapping

    @RequestMapping("login")
    public String login() {

        return "login";
    }

//    可以通过一个字符数组装入注册的地址,可以响应多个请求地址,method用于指定前端发送什么请求类型才会响应
    @RequestMapping(value = {"register","login"},method = RequestMethod.GET)
    public String register()
    {
        return "register";
    }

//    模糊匹配,*表示前面如果是/user,则后面不管是/user/dasdasdasd /user/ds都能匹配 不过多层不可匹配/user/a/a这样就不能匹配
//    **表示前面如果是/user,则后面不管是/user/aa/aaaa/aaaa/,多少层都能够匹配
//    @GetMapping表示该方法以get方式接收请求实际源码只不过是多加了一个method 指明了请求方式罢了
      @GetMapping({"/user/*","/user/**"})
    public String modifyPassword(){
        return "modifyPassword";
    }

}

2.接收参数

java 复制代码
 @RequestMapping("data")
    @ResponseBody
//    形参列表,填写对应的名称参数即可! 请求参数名 = 形式参数名即可,这样会自动赋值
    public String data(String name,int age)
    {
        System.out.println("name: "+name+"  age: "+age);
        return "name: "+name+"  age: "+age;
    }

通过指定requestMapping注释能够指定接收路径

通过ResponseBody能够指定该类直接返回给前端字符串进行接收

1.@Param注解

通过value指定传入参数时的key名字,就算是传入的key名字和参数名相同也会报错

required用于指定是否是必须被传入的参数,默认值是true代表必须传入,否则报错

2.一名多值

可以通过一个列表接收传入的多个值

但是注意,如果未使用requestParam那么就会报错

因为未使用requestParam就会代表着直接将一个字符串赋值给了一个列表报错

3.实体类接收参数

只需要将传入的参数名和实体类名保持一致即可

4.路径传参接收

要点:在路径标签中指定其路径的时候使用花括号将其括起来

然后再参数列表中声明一个与路径中花括号括起来的key值相同的名字即可

这样就可以获取到路径中的值,如下图结果所示

不过要注意,我们需要使用@PathVariable写在参数名前面,不然正常情况下默认规定其通过param方式获取数据

即便参数列表中参数名不相同依旧可以指定value的对应值与mapping注解中的key值传参

5.json数据接收

Java在没有导入对应的json转换依赖的时候是不能够获取到json数据的

我们导入一个json依赖

在配置类上方添加一个

复制代码
@EnableWebMvc

注解就可以使得适配器转化json数据并发送给类进行接收

@EnableWebMvc解释

@EnableWebMvc注解效果等同于在 XML 配置中,可以使用 `<mvc:annotation-driven>` 元素!我们来解析`<mvc:annotation-driven>`对应的解析工作!

让我们来查看下`<mvc:annotation-driven>`具体的动作!

先查看标签最终对应解析的Java类

查看解析类中具体的动作即可

打开源码:org.springframework.web.servlet.config.MvcNamespaceHandler

打开源码:org.springframework.web.servlet.config.AnnotationDrivenBeanDefinitionParser

在对应的源码中,我们可以发现他将一个Adapter(经理)加入了ioc容器,并且在Adapter对象中加入了json字符串适配器能够解析json串

在源码中由于已经将Adapter和Handler加入了ioc容器,我们就不需要在手动生成这两对象假如ioc容器中,只需要添加一个注解即可

6.接收cookie数据

先来介绍一下cookie是什么

Cookie,即"小甜饼"的意思,在计算机领域中,特指一种由服务器发送到用户浏览器并保存在用户计算机(客户端)上的小型文本文件。它满足RFC6265标准,通常用于辨别用户身份、跟踪用户活动、保存用户设置等。以下是对Cookie的详细解释:

一、Cookie的定义

Cookie是一种技术,允许网站服务器把少量数据储存到客户端的硬盘或内存,或是从客户端的硬盘读取数据。这些数据以"名/值"对(name-value pairs)的形式储存,通常经过了加密处理,因此一般用户看到的只是一些毫无意义的字母数字组合,只有服务器的CGI处理程序才知道它们真正的含义。

二、Cookie的作用

  1. 会话管理:Cookie最初也是最主要的作用就是用于会话管理。当用户登录一个网站时,服务器会生成一个包含会话ID的Cookie并发送给浏览器,浏览器将这个Cookie保存在本地。此后,每次用户发送请求时,浏览器都会自动将这个Cookie发送给服务器,服务器通过会话ID识别用户身份,从而保持用户的登录状态。
  2. 个性化设置:Cookie还可以用来保存用户的个性化设置,如主题、语言、字体大小等。这样,当用户再次访问网站时,网站可以根据Cookie中的信息为用户提供更加个性化的体验。
  3. 购物车功能:在电子商务网站中,Cookie经常被用来实现购物车功能。当用户将商品添加到购物车时,这些信息会被保存在Cookie中。这样,即使用户关闭了浏览器或换了一台电脑,只要Cookie还在,购物车中的商品信息就不会丢失。
  4. 跟踪用户行为:网站可以使用Cookie跟踪用户在网站上的行为,如访问了哪些页面、停留了多长时间、点击了哪些链接等。这有助于网站分析用户行为,优化网站设计和内容。
  5. 广告定向:除了网站自己设置的Cookie外,还有一些第三方Cookie,它们通常由广告商或数据分析公司设置。这些Cookie可以用来跟踪用户在多个网站上的行为,从而为用户提供更加精准的广告定向服务。

三、Cookie的类型

  1. 会话Cookie(Session Cookies):这种类型的Cookie在浏览器关闭后就会被删除,主要用于保存用户的会话信息。
  2. 持久Cookie(Persistent Cookies):与会话Cookie不同,持久Cookie会在用户的计算机上长期保存,直到其过期时间到达或被用户手动删除。这种类型的Cookie常用于保存用户的登录状态、个性化设置等信息。
  3. 安全Cookie(Secure Cookies):安全Cookie只能通过HTTPS协议传输,不能通过未加密的HTTP协议传输。这增加了Cookie在传输过程中的安全性。
  4. HttpOnly Cookie:HttpOnly是一个标志属性,用于防止JavaScript代码访问特定的Cookie。当设置了HttpOnly属性的Cookie被创建后,它将无法通过客户端脚本(如JavaScript)进行访问。这有助于减少跨站脚本攻击(XSS)的风险。

四、Cookie的安全性问题及防范措施

虽然Cookie在许多方面都非常有用,但它们也存在一些潜在的安全风险,如XSS攻击、CSRF攻击、Cookie劫持和隐私泄露等。为了防范这些风险,网站可以采取以下措施:

  1. 设置HttpOnly属性、对输出进行编码等,以防止XSS攻击。
  2. 使用Token验证、检查请求的来源等,以防止CSRF攻击。
  3. 使用HTTPS协议对Cookie进行加密传输、设置SameSite属性等,以防止Cookie劫持。
  4. 遵循最小必要原则收集和使用用户数据,并采取加密、匿名化等安全措施,以保护用户隐私。

综上所述,Cookie作为一种重要的客户端技术,在互联网应用中发挥着举足轻重的作用。然而,在使用Cookie的过程中,我们也需要注意其潜在的安全风险,并采取相应的措施进行防范和保护。

那么我们如何接收cookie数据呢

我们在Java类中编写如下一个类,在参数中通过cookievalue指定cookiename的请求的参数名字

我们先用save通过响应报文存储一个cookie数据

可以通过图片看到存储成功了

接着我们通过cookiename这个key值从cookie中取出对应的value值显示在屏幕上

7.接收请求头数据

通过RequestHeader标签写在参数上,就能够通过参数获取到对应请求头的中的key值为Host的数据

8.获取原生api对象

SpringMVC接收数据总结

SpringMVC响应数据(输出数据)

两种开发模式介绍

1.前后端不分离介绍

整体调用流程

1.浏览器通过向服务端发送请求进行一些数据交互和页面跳转请求

2.controller层接收后通过三层架构一步步发送到mapper层(Dao层)

3.dao层通过和数据库交互获取数据返回到controller进行数据封装处理

4.在前后端不分离项目中,controller会将数据放入共享域中,并且找到需要跳转的html页面

5.然后前端页面会从共享域中拿取数据,并且将模板页面发送给controller层

6.接着将controller层中返回到的html页面发送给浏览器

7.主要用于小型的管理系统的开发

开发流程

我们首先在webapp下面的web-inf的文件中声明一个.jsp文件这是一个网页

接着我们创建一个配置类,让他继承WebMvcConfigurer

继承这个类之后我们就可以将视图解析器装入ioc容器中了(该类还封装了很多mvc容器需要的类,视图解析器的作用是将.jsp文件进行解析)

我们在重写的方法中,通过registry将文件的index.jsp的文件进行前后缀注册,以便能够定位到视图的位置

接着我们创建一个jspcontroller能够将视图返回给前端

其注意要点图里面就有

其中setAttribute是将数据放入共享数据域的

观察jsp文件,在共享数据区域获取数据

这是返回的结果

转发与重定向实现

转发指的是访问项目内的文件,范围只能是项目内

重定向指可以是向内也可以是向外

其在springMVC中的实现如下

先观察一下资源文件的访问路径是/jsp/index

转发

在转发中,我们先指定转发的访问路径将其设置为forward

然后我们在方法内返回视图文件的访问路径即可,但是要注意

在访问路径前面需要加一个forward编译器才能判定他是在进行转发

否则会直接识别为这是资源文件的位置进行解析,就如开发流程那里所说的一眼

通过转发的方式我们也到达了forward的文件的位置

重定向

其注意要点如图所示

这是向内重定向

同样也有向外重定向

2.前后端分离模式

主要通过json和前端进行数据交互

前端都是调用的一个后端接口获取数据

一般用于大型商业部署项目开发,多端部署

返回json串

其原理如下图所示

核心的注解是一个@ResponeseBody

通过responesebody能够告知adapter我们即将返回的是一个json串

然后让adapater进行处理,将Java类对象转化为一个json串发送给前端

通过使用responseBody能够使得返回视图和转发重定向的语法全部生效

具体controller类的编写如下

通过controller添加到ioc容器

通过requestmapping指定访问路径

接着在需要返回json串的类/方法上添加@ResponseBody即可返回一个json串

也可以是返回一个list对象

这两个方法的返回结果如下图所示

RestController注解

可以观察源码,同时包含了controller和responsebody注解

能够使用这一个注解替代controller和responsebody注解

返回静态资源

我们现在文件目录下面放入一个静态图片资源

我们在平常的客户端请求文件的路径时是访问不到的

因为默认情况下当服务端接收请求之后,回去handlerMapping()中寻找有没有对应的handler请求路径

此时会发现没有对应的请求路径,那么就会发生错误

那么我们如何解决这种情况呢

在配置类中重写一个如红圈所示的接口就可以实现找到静态资源

那么其原理是什么呢

我们点进去接口里面进行观察,如红箭头所指的类是返回静态资源得以成功的方法

接着点进去那个添加Servlet进行观察

我们可以发现,他添加了一个handler对静态资源进行处理

那么这个handler是如何对静态资源进行处理的呢

在这个handler是这样处理的,假如我们没有找到路径,那么就会对图片资源的路径进行内部转发

通过转发去寻找请求地址所需要的资源

原理图

Servlet接收到一个请求,先去handlerMapping寻找有没有对应的controller类查找数据

发现没有,那么handlerMapping就会找DefaultServletHandler去转发图片的路径寻找资源

结果如下

RESTFul风格

一种基于http协议的规范编码风格

其请求路径的设计如下图所示

全局异常处理机制

什么是声明式异常和编程式异常?

编程式:在代码内部有详细的异常捕捉

声明式:在外部使用一个配置文件对需要进行异常捕获的文件进行包裹,有异常时就会调用外部异常处理代码进行处理

全局异常处理注释

所有需要的异常都在图中写清楚了

详解再图片下方写

@ControllerAdvice和@RestControllerAdvice解析

ControllerAdvice:标识这是一个全局异常处理类,当全局异常发生时就会寻找该类进行处理,可以返回逻辑视图和转发重定向

RestControllerAdvice:点进去后其实可以发现源代码里面有controlleradvice和responsebody注解在里面

responseBody是告知这个类会返回一个json字符串

@ExceptionHandler

异常处理的类,在其后面的括号内的参数是用于指定该方法处理什么类型的异常

拦截器

类似于javaweb中的Filter(过滤器)

对一些特殊的请求进行拦截处理

拦截器 Springmvc VS 过滤器 javaWeb:

  • 相似点

  • 拦截:必须先把请求拦住,才能执行后续操作

  • 过滤:拦截器或过滤器存在的意义就是对请求进行统一处理

  • 放行:对请求执行了必要操作后,放请求过去,让它访问原本想要访问的资源

  • 不同点

  • 工作平台不同

  • 过滤器工作在 Servlet 容器中

  • 拦截器工作在 SpringMVC 的基础上

  • 拦截的范围

  • 过滤器:能够拦截到的最大范围是整个 Web 应用

  • 拦截器:能够拦截到的最大范围是整个 SpringMVC 负责的请求

  • IOC 容器支持

  • 过滤器:想得到 IOC 容器需要调用专门的工具方法,是间接的

  • 拦截器:它自己就在 IOC 容器中,所以可以直接从 IOC 容器中装配组件,也就是可以直接得到 IOC 容器的支持

其作用范围如图所示

因为使用了springmvc框架,那么在程序内部是只有一个DispatcherServlet进行请求处理的

假如使用filter的话,只能在Servlet外部即整个springMVC框架外部进行拦截处理,一刀切,并不能在里面的各种资源调用的内部进一些业务逻辑的处理

那么使用拦截器就可以解决这一问题

拦截器作用的位置

拦截器如何使用

首先在配置类中注册一下拦截器(注意,图中的这种注册方式是拦截全部请求的,至于具体到方法的后续会讲解到)

拦截器类的书写,我们需要继承一个HanlerInterceptor类

其需要重写如下三个方法

preHandle

postHandle

afterCompletion

preHandle方法

其参数和注意事项如下

其中handler就是我们进行拦截的目标方法

preHandler是在目标方法执行前执行的代码

postHandle方法

其参数和主义的事项如下

其中modelAndView返回的是一个视图或者共享域对象

不需要时为空

afterCompletion方法

在前面两个方法执行完毕后调用

其中ex参数是当目标方法发生异常时传入,并进行输出

拦截器的指定拦截

链式调用

在将拦截器类进行注册后添加一个addpathPatterns就可以进行拦截

排除拦截

在前一个大的路径内,指定一个在大路径内的小路径就可以进行小路径的拦截

拦截的顺序

根据注册的先后进行拦截,其源码其实是所有的拦截类放入一个数组

然后prehandler是正序遍历数组进行方法调用,所以先进入的类会先执行

接着post和after方法是逆序遍历,也就是说,先进入的类后遍历

源码

通过mapping(秘书)对对应的拦截方法进行获取

观察下面的代码,下方红箭头所指通过handler获取拦截方法执行的目标方法

在红箭头的上方,有一个拦截方法,他会判断前置prehadler的返回结果,假如是一个false那么就进入方法执行return代表着剩下的目标方法的逻辑不执行,实现了拦截

参数校验

可以通过给实体类的变量上添加注解,不符合对应的要求的就会报错

在对应的实体类中添加对应的注解

接着我们需要在对应的接收前端数据的controller层接收对应实体类参数的方法中打的参数上添加

@Validdated注解!!!

如果不添加那自动校验就不会生效

以字符串不能为空举例当字符串为空时,会将报错信息默认的返回

但是我们发现这种方法前端不容易读懂,所以我们需要自定义的定义报错信息的返回

我们需要定义一个BindingResult类进行错误的接收

注意,该类一定要在参数列表中紧挨着需要校验的实体类

接着我们通过hasErrors判断是否报错

如果报错,我们定义一个集合将错误参数放入并返回

总结

调用流程,很关键

相关推荐
江木12318 分钟前
CUDA C 编程入门学习记录
c语言·开发语言·学习
鹿屿二向箔1 小时前
搭建一个基于Spring Boot的书籍学习平台
spring boot·后端·学习
APItesterCris1 小时前
如何监控和防范小红书笔记详情API的安全风险?
网络·笔记·安全
有梦想有行动2 小时前
kafka学习
学习
bohu832 小时前
opencv笔记1
人工智能·笔记·opencv
三月七(爱看动漫的程序员)3 小时前
Active Prompting with Chain-of-Thought for Large Language Models
数据库·人工智能·深度学习·学习·语言模型·自然语言处理
半路程序员3 小时前
kubernetes学习-Service(七)
学习·容器·kubernetes
Naion4 小时前
统计学习算法——逻辑斯谛回归
学习·算法·回归
123yhy传奇4 小时前
【学习总结|DAY032】后端Web实战:登录认证
java·spring boot·学习·mybatis
fanged5 小时前
LDD3学习9--数据类型和定时器
linux·学习