Spring MVC学习随笔-控制器(Controller)开发详解:控制器跳转与作用域(二)视图模板、静态资源访问

news2025/1/16 14:02:55

学习视频:孙哥说SpringMVC:结合Thymeleaf,重塑你的MVC世界!|前所未有的Web开发探索之旅

衔接上文Spring MVC学习随笔-控制器(Controller)开发详解:控制器跳转与作用域(一)

  • SpingMVC中request作用域的处理

    • 代码

      # 基于Model的方式 
      @RequestMapping("view2")
      public String view2(Model model) {
      		// 等同于 request.addAttribute();
          model.addAttribute("name", "suns");
          return "result1";
      }
      # 基于ModelMap的方式 同上
      @RequestMapping("view3")
      public String view3(ModelMap modelMap) {
          modelMap.addAttribute("name", "suns2");
          return "result1";
      }
      
    • Model、ModelMap相关细节分析

      1. 通过Model、ModelMap进行作用域处理,可以解决视图模板技术耦合的问题

        因为SpringMVC通过视图解析器区别JSP、FreeMaker,再将Model、ModelMap的数据放到request或root里面运行。

      2. SpringMVC中提供Model和ModelMap两种方式处理request作用域,他们的区别是什么

        虽然两者表现形式以及声明的形参类型都不同,但是在运行时,SpringMVC会动态提供对应的实现类型,名字是BindingAwareModelMap。所以本质上两者相同。

      3. 为什么不直接使用BindingAwareModelMap?

        在源码中,Model接口会根据开发者使用SpringMVC或Spring WebFlux进行自动适配,使用MVC开发时会使用BindingAwareModelMap,而WebFlux开发时使用的是ConcurrentModel。

      4. SpringMVC为什么会提供两种开发方式?Model、ModelMap这两种开发方式,更推荐哪种使用?

        推荐使用Model,ModelMap是老系统使用的,为了兼容老系统所以留着。Model可以兼容传统MVC也可以在WebFlux中使用,更有利于项目维护。

      5. 如果redirect跳转,数据如何跳转

        SpringMVC会自动把Model或ModelMap中的数据,通过?的形式在url上进行拼接,从而传递数据

  • SpringMVC中Session作用域的处理

    • 基本使用方式及其存在的问题

      session.setAttribute("name","value");
      
      @RequestMapping("view")
      public String view1(HttpSession session){
      		session.setAttribute("name","value");
      		return "result";
      }
      存在问题:与ServletAPI耦合,在SpringMVC中不建议使用。
      
    • @SessionAttributes注解

      • 存储数据

        @Controller
        @RequestMapping("view3")
        **@SessionAttributes(value = "name")  // 声明name存在session作用域**
        public class View3Controller {
        
            @RequestMapping("view1")
            public String view1(Model model) {
                model.addAttribute("name", "xiaojr");
        				model.addAttribute("age", 10);
                return "result1";
            }
        }
        // jsp中取值
        <html>
        <body>
            <h1>session attribute is ${sessionScope.name}</h1>
            <h1>request attribute is ${requestScope.age}</h1>
        </body>
        </html>
        
      • 注意

        1. Model、ModelMap把name的数据通过@SessionAttributes存储在Session作用域中的同时在Request作用域中也会存储。
        2. 此时Request作用域、Session作用域存储的是一个对象的引用。
    • 删除Session作用域中的数据

  • SpringMVC中application作用域的处理

  • 为什么SpringMVC没有提供application作用域

    因为application这个作用域是全局唯一,在开发中多用于存储全局唯一的对象,被框架底层封装,在开发时,程序员基本不会使用其用于业务操作。

  • @ModelAttribute注解

    1. SpringMVC通过@ModelAttribute注解:接受请求参数的同时,把数据存储到request作用域中。

      @RequestMapping("view1")
      public String view1(@ModelAttribute("name") String name) {
          System.out.println("View4Controller.view1");
          return "result2";
      }
      
    • 使用场景


      传递页码时可以使用。

    • 注意细节

      • 细节一

        如果传递的是简单变量参数:则value属性必须与超级链接或表单的key名称保持一致。

        如果传递的是POJO类型的请求参数:则没有上述要求,但是value属性会作为request作用域的名称。

      • 细节二

        1. @ModelAttribute中value的值不能与@SessionAttributes中value的值一致,如果一致则会产生异常
        2. 如果要把请求参数的数据也存在session作用域中,需要将request作用域存储改为使用传统做法:model.addAttribute(”name”,name);
  • 什么是ModelAndView【了解】

    • 什么是ModelAndView

      ModelAndView这个类型,实际上是一个复合类型,起到了2个方面的作用。

      注意:ModelAndView里面只能使用ModelMap,这也是它的劣势之一。

      1. Model 代表作用域的操作,就是前面的ModelMap。
      2. View 代表跳转路径(页面),对应前面的四种跳转。

      最终这两方面的工作统一被ModelAndView进行封装,做为控制器方法的返回值使用。

    • 总结目前控制器方法返回值

      1. public String controller();
      2. public ModelAndView controller();
      注意:SpringMVC处理跳转页面和作用域时,把相应的内容都会封装到ModelAndView中,
      所以ModelAndView返回值的这种处理更加底层,而返回值String的处理仅是简化开发。
      

5.5 视图控制器

5.5.1 什么是视图控制器

  1. 视图控制器可以通过配置的方式,访问受保护的视图模板,简化开发。

  2. 什么是视图模板?

    JSP Thymeleaf FreeMarker Velocity

  3. 为什么需要保护视图模板?

    目前的开发方式都没有对视图模板进行保护,有可能导致程序在被用户访问时产生非预期效果(Bug)

  4. 如何保护视图模板

    将所有视图模板放置在WEB-INF下,这样用户就无法通过地址直接访问视图模板了。

5.5.1 受保护的视图模板如何访问?

  1. 所有的视图模板,只能通过控制器forward访问:return "result"; return "forward:/WEB-INF/jsp/result.jsp";

    记得同时修改viewResolver

    <bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <property name="prefix" **value="/WEB-INF/jsp/"/>**
        <property name="suffix" value=".jsp"/>
    </bean>
    

5.5.2 视图控制器

虽然可以通过控制器forward访问视图模板,但是直接访问页面也需要控制器,因此会变得繁琐,所以有了视图控制器的存在:通过在配置文件(dispatcher.xml)可以直接访问

<mvc:view-controller path="/result3" view-name="result3"/>
注意:path不能和@RequestMapping的路径冲突

5.5.3 视图控制器的Redirect跳转

@RequestMapping("view1")
public String view1() {
    System.out.println("View6Controller.view1");
    return "redirect:/result4";
}

<mvc:view-controller path="/result4" view-name="result4"/>

5.6 静态资源处理

5.5.1 什么是静态资源

  1. 所谓静态资源,指的是项目中非java代码部分的内容,如 图片、js文件、css文件。

  1. 目前SpringMVC的开发中,按照现有的配置内容,是无法访问静态资源的。


g)

在SpringMVC底层,资源请求访问DispatcherServlet后,会将其请求当成控制器并调用对应的控制器方法创建对象,但是在静态资源访问,第一步就走不通了,所以就报错404。

5.5.3 解决方式

  • 方式一【DefaultServlet】

    Tomcat提供了能够访问静态资源的DefaultServlet(web.xml)

    在web.xml添加DefaultServlet后的代码:

  • 方式二【default-servlet-handler】推荐

    • 第一种开发方式的问题

  • default-servlet-handler开发方式

    1. 在SpringMVC的配置文件(dispatcher.xml)中,配置<mvc:default-servlet-handler/>即可
  • 实现原理

    1. <mvc:default-servlet-handler/>标签的底层也是调用了defaultServlet进行的静态资源处理。

    2. <mvc:default-servlet-handler/>标签他是如何调用defaultServlet进行静态资源处理的?

      他的底层是通过DefaultServletHttpRequestHandler,以forward的形式调用的DefaultServlet。

    mvc:default-servlet-handler/完整运行流程:用户如果发起控制器的请求,首先会到DispatcherServlet,然后DispatcherServlet会通过RequestMappingHandlerMapiping查找控制器上@RequestMapping注解并进行相应的匹配,进而查找,如果查找到则会通过HandlerAdapter进行处理。对于静态资源的操作,也就是使用了<mvc:default-servlet-handler/>的处理,它的请求也一样会访问DispatcherServlet,没查找到则会通过SimpleUrlHandlerMapping调用DefaultServletHttpRequestHandler进而调用DefaultServlet进行静态资源的处理。

下一章:Spring MVC学习随笔-文件下载和上传(配置文件上传解析器multipartResolver)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1281077.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

网络和Linux网络_9(应用层和传输层_笔试选择题)

目录 一. 常见应用协议等等 1. 以下不是合法HTTP请求方法的是( ) 2. 文件传输使用的协议是&#xff08;&#xff09; 3. HTTP1.1的请求方法不包括&#xff1f;() 4. http状态码中&#xff0c;( )表示访问成功&#xff0c;( )表示坏请求&#xff0c;( )表示服务不可用。() …

Visual Studio 2022+Python3.11实现C++调用python接口

大家好&#xff01;我是编码小哥&#xff0c;欢迎关注&#xff0c;持续分享更多实用的编程经验和开发技巧&#xff0c;共同进步。 查了一些资料&#xff0c;不是报这个错&#xff0c;就是报哪个错&#xff0c;没有找到和我安装的环境的一致的案例&#xff0c;于是将自己的摸索分…

冲突域和广播域

文章目录 冲突域广播域 冲突域 在网络内部两个数据帧同时进行传输时&#xff0c;产生与发生冲突的区域&#xff0c;所有共享介质都是一个冲突域。冲突域时基于第一层&#xff0c;物理层的。 集线器和中继器因为都在物理层&#xff0c;没有MAC地址表&#xff0c;所以不能隔离冲…

Spring Cloud Alibaba简介

1、简介 Spring Cloud阿里(https://sca.aliyun.com/en-us/)为分布式应用开发提供一站式解决方案。它包含开发分布式应用程序所需的所有组件&#xff0c;使您可以轻松地使用Spring Cloud开发应用程序。 有了Spring Cloud阿里&#xff0c;你只需要添加一些注释和少量的配置&#…

MDK提示:在多字节的目标代码中,没有此Unicode 字符可以映射到的字符

MDK警告提示在多字节的目标代码中&#xff0c;没有此Unicode 字符可以映射到的字符 警告提示&#xff1a; 在写MDK的工程代码时&#xff0c;发现代码中引入的头文件前方出现一些红色的叉叉&#xff0c;但是编译工程并不报错&#xff0c;功能也能正常执行的&#xff0c;只是提…

mysql中除了InnoDB以外的其它存储引擎

参考资料&#xff1a;https://dev.mysql.com/doc/refman/8.0/en/storage-engines.html MyISAM存储引擎 https://dev.mysql.com/doc/refman/8.0/en/myisam-storage-engine.html MyISAM 存储引擎是基于比较老的ISAM存储引擎&#xff08;ISAM已经不再可用&#xff09;&#xff…

android studio安装SDK时无法勾选

这两天帮助学妹安装android studio安装SDK时无法勾选&#xff0c;记录一下最终解决办法。头大。 核心 360 问题 网上所有方法都尝试了包括挂梯子&#xff0c;改hosts&#xff0c;盘符权限等等。 最终解决下载360 使用这两个&#xff0c;DNS注意要用8.8.8.8的 成功解决

DevEco Studio设置背景图片

我们打开编辑器 左上角菜单 选择 File 下的 Settings 我们选择首选项 Appearance 8 Behavior 下的 Appearance 右侧界面 点击下面的 Background lmage… 新弹出的这个窗口 我们可以鼠标 拖拽它的边 把他拉大一点 当前 我们代码中是没有背景图片的 我们点击入下图指向的三个…

基于Java SSM框架+Vue实现药品保健品购物网站项目【项目源码+论文说明】

基于java的SSM框架Vue实现药品保健品购物网站演示 摘要 随着社会的发展&#xff0c;社会的各行各业都在利用信息化时代的优势。计算机的优势和普及使得各种信息系统的开发成为必需。 ssm药源购物网站&#xff0c;主要的模块包括两个用户&#xff0c;管理员权限&#xff1a;用…

Kubernetes基础(十)-自动伸缩

1 介绍 Kubernetes提供了多种自动伸缩机制&#xff0c;主要常见的有&#xff1a; HPA&#xff08;Horizontal Pod Autoscaling&#xff09;VPA&#xff08;Vertical Pod Autoscaler&#xff09;CA&#xff08;Cluster Autoscaler&#xff09;CPA&#xff08;Custom Pod Autos…

【c】求一组数据的最大值和第二大的值

我们可以创建数组&#xff0c;利用冒泡排序法把数组进行排序&#xff0c;但是当元素过多时候循环可能过多导致循环超限 所以我们可以换种其他方法&#xff0c;代码附上 #include<stdio.h> int main() {int n,i;puts("输入这组数据的个数");scanf("%d&qu…

Python优化利器:Numba库深度探究

更多资料获取 &#x1f4da; 个人网站&#xff1a;ipengtao.com Numba 是一个用于优化 Python 代码的开源即时编译器&#xff0c;能够将 Python 代码转换为本机机器码&#xff0c;提高其执行速度。其主要特点包括&#xff1a; 能够加速整数、浮点数等数值计算。支持直接在 CP…

【涨薪技术】深入接口测试之Mock技术

01、为什么要用Mock 服务端与客户端约定了接口&#xff0c;但服务端还没有完成开发时&#xff0c;客户端一般由如下处理方式&#xff1a; 1、在程序中写模拟数据 程序中增加垃圾代码&#xff0c;后期还要删除&#xff0c;可能对代码造成影响模拟异步请求不方便服务端接口开发…

Ubuntu20.04安装ROS2

官方参考文章 Ubuntu (Debian) — ROS 2 Documentation: Foxy documentation curl密钥问题 sudo curl -sSL https://raw.githubusercontent.com/ros/rosdistro/master/ros.key -o /usr/share/keyrings/ros-archive-keyring.gpg curl: (7) Failed to connect to raw.githubus…

Docker下安装MySQL

如果在Docker下直接拉取MySQL并运行镜像&#xff0c;由于没有指定字符编码集&#xff0c;可能会存在插入中文出现乱码的情况&#xff0c;并且当容器删除后&#xff0c;容器里面存在的数据会丢失&#xff0c;所以在运行容器时应该使用数据卷进行挂载&#xff0c;按照如下步骤操作…

31-WEB漏洞-文件操作之文件包含漏洞全解

31-WEB漏洞-文件操作之文件包含漏洞全解 一、本地包含1.1、无限制包含漏洞文件1.2、有限制包含漏洞文件1.2.1、绕过方法1.2.1.1、%00截断1.2.1.2、长度截断 二、远程包含2.1、无限制包含漏洞文件2.2、有限制包含漏洞文件 三、各种协议流提交流3.1、各协议的利用条件和方法3.1.1…

【Windows】如何实现 Windows 上面的C盘默认文件夹的完美迁移

如何实现 Windows 上面的C盘默认文件夹的完美迁移 1. 遇到的问题 在我想迁移C盘的 下载 和 视频 文件夹的时候&#xff0c;遇到了这样的问题&#xff0c;在迁移之后&#xff0c;我显卡录像的视频还是保存到了C盘默认位置里&#xff0c;以及我迁移了 下载 之后下载的盘依然是在…

XSS漏洞原理

XSS漏洞介绍&#xff1a; 跨站脚本攻击XSS(Cross Site Scripting)&#xff0c;为了不和层叠样式表(Cascading Style Sheets, CSS)的缩写混淆&#xff0c;故将跨站脚本攻击缩写为XSS。恶意攻击者往Web页面里插入恶意Script代码&#xff0c;当用户浏览该页面时&#xff0c;嵌入We…

数据结构 | 查漏补缺之ASL、

目录 ASL 情形之一&#xff1a;二分查找 线索二叉树 哈夫曼树 大根堆 邻接表&邻接矩阵 ASL 参考博文 关于ASL(平均查找长度)的简单总结_平均查找长度asl-CSDN博客 情形之一&#xff1a;二分查找 线索二叉树 参考博文 线索二叉树(线索链表遍历&#xff0c;二叉树…

【拓展】Loguru:更为优雅、简洁的Python 日志管理模块

目录 一、简单介绍 二、安装与简单使用 ​三、常见用法 3.1 显示格式 3.2 写入文件 3.3 json日志 3.4 日志绕接 3.5 并发安全 四、高级用法 4.1 接管标准日志logging 4.2 输出日志到网络服务器 4.2.1 自定义日志服务器 ​4.2.2 第三方库日志服务器 4.3 与pytest结…