当前位置: 首页 > news >正文

Spring MVC Session 属性 (@SessionAttributes) 是什么?如何使用它共享数据?

在 Spring MVC 中,@SessionAttributes 是一个类级别的注解,用于表示 Spring MVC 将 Controller 中的属性存储到 HTTP Session 中,以便在同一用户会话的后续请求中可以访问这些属性。

简单来说,它的作用是将请求范围的模型(Model)属性“提升”到会话范围。

为什么需要 @SessionAttributes

模型属性(通过 Model, ModelMap, ModelAndView 添加的)只在当前请求的生命周期内有效。请求处理完成后,模型数据就会丢失。

但在某些场景下,我们需要在同一个用户的多个连续请求之间共享数据,最典型的例子就是多步骤表单 (Wizard Form)。用户在第一页填写一部分信息提交,然后跳转到第二页填写更多信息,最后在第三页或最终页将所有信息一起提交保存。此时,从第一页和第二页收集到的数据需要在整个过程中保持可用。

虽然可以直接使用 HttpSession 对象来存取数据,但 @SessionAttributes 提供了一种与 Spring MVC 的模型和请求处理机制更紧密集成的方式来管理会话范围的数据,特别是与 @ModelAttribute 结合使用时非常方便。

@SessionAttributes 的工作原理:

  1. 标注: 在 Controller 类上使用 @SessionAttributes 注解,指定一个或多个模型属性的名称(或类型)。
  2. 存储: 当 Controller 方法将一个属性添加到模型中,并且该属性的名称(或值的类型)与 @SessionAttributes 中指定的匹配时,Spring MVC 会在请求处理完成后,自动将这个模型属性存储到当前用户的 HTTP Session 中。
  3. 检索: 对于后续到达同一个 Controller 的请求,在请求处理方法执行之前,Spring MVC 会检查 HTTP Session 中是否存在与 @SessionAttributes 中指定的名称(或类型)匹配的属性。如果存在,Spring 会自动将这些属性从 Session 中加载回当前请求的模型中。
  4. 绑定 (@ModelAttribute): 这与自动绑定的过程结合得非常好。如果有一个方法参数使用 @ModelAttribute 标注,并且该参数的名称(或类型)与 @SessionAttributes 中指定的匹配,Spring 会尝试从 Session 中获取对应的对象,而不是从请求参数中新建一个对象。然后,它会根据请求参数更新这个从 Session 中取出的对象。

如何使用 @SessionAttributes

@SessionAttributes 可以通过两种方式指定要存储的属性:

  • 按名称指定: 使用 value 或其别名指定模型属性的名称。
  • 按类型指定: 使用 types 指定模型属性的类型。任何添加到模型中的对象,如果其类型匹配,就会被存储。

示例:一个简单的多步骤表单

假设我们有一个创建用户的三步流程。我们在 Controller 中使用 @SessionAttributes 来保存用户对象 UserForm

首先,定义一个 POJO (UserForm) 来封装表单数据:

public class UserForm implements Serializable { // 建议实现 Serializableprivate String firstName;private String lastName;private int age;// Getters and setterspublic String getFirstName() { return firstName; }public void setFirstName(String firstName) { this.firstName = firstName; }public String getLastName() { return lastName; }public void setLastName(String lastName) { this.lastName = lastName; }public int getAge() { return age; }public void setAge(int age) { this.age = age; }@Overridepublic String toString() {return "UserForm{" +"firstName='" + firstName + '\'' +", lastName='" + lastName + '\'' +", age=" + age +'}';}
}

然后,创建 Controller:

import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.ModelAttribute;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.SessionAttributes;
import org.springframework.web.bind.support.SessionStatus;@Controller
@RequestMapping("/user/create")
// 1. 使用 @SessionAttributes 标注 Controller 类
//    指定要将名为 "userForm" 的模型属性存储到 Session 中
@SessionAttributes("userForm")
public class UserCreationController {// 2. @ModelAttribute 方法:用于在模型中初始化 userForm 对象//    这个方法会在任何请求处理方法执行前被调用一次 (通常是第一次访问该Controller时)//    返回的对象会以 "userForm" 为key添加到模型中 (因为方法名是userForm,或者可以用@ModelAttribute("userForm")指定)//    因为 @SessionAttributes("userForm"),这个对象会被存入 Session@ModelAttribute("userForm")public UserForm setUpUserForm() {return new UserForm();}// Step 1: 显示第一个表单页面@GetMapping("/step1")public String step1(Model model) {// @ModelAttribute("userForm") setUpUserForm() 方法已经把 userForm 加入模型// 这里可以直接返回视图名return "user/create/step1";}// Step 1: 处理第一个表单提交// 3. 使用 @ModelAttribute("userForm") 参数//    Spring 会从 Session 中加载 userForm (如果存在),然后将请求参数绑定到这个对象上@PostMapping("/step1")public String processStep1(@ModelAttribute("userForm") UserForm userForm) {System.out.println("Processed Step 1: " + userForm);// 跳转到下一步return "redirect:/user/create/step2";}// Step 2: 显示第二个表单页面@GetMapping("/step2")public String step2(@ModelAttribute("userForm") UserForm userForm) {// userForm 自动从 Session 加载并添加到模型return "user/create/step2";}// Step 2: 处理第二个表单提交@PostMapping("/step2")public String processStep2(@ModelAttribute("userForm") UserForm userForm) {System.out.println("Processed Step 2: " + userForm);// 跳转到下一步return "redirect:/user/create/step3";}// Step 3: 显示最终确认页面@GetMapping("/step3")public String step3(@ModelAttribute("userForm") UserForm userForm) {// userForm 自动从 Session 加载并添加到模型return "user/create/step3";}// Step 3: 处理最终提交并完成流程// 4. 在最终处理方法中,使用 SessionStatus 参数//    调用 setComplete() 方法清理 Session 中的 @SessionAttributes 属性@PostMapping("/complete")public String complete(@ModelAttribute("userForm") UserForm userForm, SessionStatus status) {System.out.println("Completing User Creation: " + userForm);// TODO: Save the userForm data to database etc.// 清理 Session 中的 "userForm" 属性status.setComplete();// 跳转到完成页面或首页return "redirect:/user/create/success";}@GetMapping("/success")public String success() {return "user/create/success";}
}

关键点解释:

  • @SessionAttributes("userForm"): 告诉 Spring 在处理完 Controller 方法后,将模型中名为 “userForm” 的属性存入 Session。在处理方法执行前,如果 Session 中有 “userForm”,会加载到模型中。
  • @ModelAttribute("userForm") public UserForm setUpUserForm(): 这个方法在第一次访问 /user/create/* 路径时会执行(或者如果 Session 中没有 “userForm” 时)。它创建一个新的 UserForm 对象并添加到模型中。
  • @ModelAttribute("userForm") UserForm userForm (作为方法参数): 在 processStep1, step2, processStep2, step3, complete 方法中,Spring 会因为 @ModelAttribute 注解尝试获取名为 “userForm” 的对象。由于 @SessionAttributes 的存在,它会首先去 Session 中查找。如果找到,就使用 Session 中的对象,并尝试将请求参数绑定到这个现有对象上。如果没有找到(通常是第一次访问 Step 1),它会调用 @ModelAttribute 方法(setUpUserForm)来创建对象。
  • SessionStatus status: 在流程的最后一步(complete 方法),我们注入了 SessionStatus 参数。调用 status.setComplete() 非常重要!它会通知 Spring 清理当前 Controller 标记为 @SessionAttributes 的所有属性。如果忘记这一步,“userForm” 对象将一直留在用户的 Session 中,直到 Session 过期,可能导致数据混乱或内存泄漏。

优点:

  • 简化了多步骤流程中的数据传递,无需手动操作 HttpSession
  • @ModelAttribute 集成,自动将请求参数绑定到 Session 中的对象。
  • 将与特定 Controller 相关的会话数据管理集中在该 Controller 中。

缺点/注意事项:

  • 必须手动清理 Session: 容易忘记调用 SessionStatus.setComplete()
  • 范围: @SessionAttributes 是类级别的,影响该 Controller 的所有请求处理方法。如果一个 Controller 处理多个不相关的流程,这可能导致问题。建议将 @SessionAttributes 用于专门处理特定工作流的 Controller。
  • Session 大小: 将大量数据或大型对象存储在 Session 中会占用服务器内存,可能影响应用的可伸缩性,尤其是在需要 Session 复制的分布式环境中。谨慎使用。
  • 并发问题: 如果同一个用户在多个浏览器标签页中同时进行同一个多步骤流程,可能会出现数据混乱。

总而言之,@SessionAttributes 是 Spring MVC 为简化特定场景(如多步骤表单)中的会话数据管理而提供的便捷机制,通过将模型属性自动在请求和会话之间同步来实现。使用时务必注意流程结束时的 Session 清理。

相关文章:

  • 信赖域策略优化TRPO算法详解:python从零实现
  • .net/C#进程间通信技术方案总结
  • 机器学习与深度学习的区别与联系:多角度详细分析
  • Linux基础(关于进程相关命令)
  • CSS opacity
  • 计算人声录音后电平的大小(dB SPL->dBFS)
  • 访问网站提示“不安全”“有风险”怎么办?
  • 3D桌面可视化开发平台HOOPS Native Platform,如何实现3D系统快速开发与部署?
  • 【网安播报】Meta 推出 LlamaFirewall开源框架以阻止 AI 越狱、注入和不安全代码
  • 【python】 python拆包
  • buck和boost总结
  • web 自动化之 selenium 元素四大操作三大切换等待
  • Oracle非归档模式遇到文件损坏怎么办?
  • AI NeRF:Meta实现手机端实时毛发渲染,开启移动AR/VR新纪元——从实验室到口袋:NeRF技术的轻量化革命
  • Android Handler 机制面试总结
  • 更新 / 安装 Nvidia Driver 驱动 - Ubuntu - 2
  • Shiro(八):JWT介绍
  • BP神经网络
  • c++:双向链表容器(std::list)
  • 【CTFSHOW_Web入门】命令执行
  • “80后”赵亮出任上海普陀区委副书记
  • 数说母亲节|妈妈的妈妈带娃比例提升,托举效果如何?
  • 央行:将支持资本市场两项货币政策工具的额度合并使用
  • 中国驻俄大使张汉晖人民日报撰文:共襄和平伟业,续谱友谊新篇
  • 巴基斯坦:印度向巴3处地点发射导弹
  • 科技赋能文化体验,“五一”假期“海昏侯”人气创新高