news 2026/4/2 0:32:08

基于 Spring Boot 的 Web 三大核心交互案例精讲

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
基于 Spring Boot 的 Web 三大核心交互案例精讲

基于 Spring Boot 的 Web 三大核心交互案例精讲
(2026年最实用写法 · 企业真实场景)

在 Spring Boot Web 开发中,真正决定项目质量和维护难度的,往往不是写了多少 Controller,而是你是否真正掌握了以下三大核心交互场景正确、优雅、可维护的处理方式:

  1. 前后端分离的登录 + Token 认证(JWT + 无感刷新)
  2. 文件上传下载(大文件、分片上传、断点续传)
  3. 实时消息推送(WebSocket + SSE 对比实战)

下面用最现代的 Spring Boot 3.x + Spring Security 6.x 写法,给你最实用的完整案例。

1. 前后端分离登录 + Token 认证(JWT + 无感刷新)

核心要点

  • Access Token 短效(15~60分钟)
  • Refresh Token 长效(7~30天),HttpOnly Cookie 存储
  • 前端拦截 401 自动刷新
  • 滚动刷新(可选,安全性更高)

后端关键代码片段(简化版)

// 登录返回双 token@PostMapping("/api/auth/login")publicResponseEntity<?>login(@RequestBodyLoginRequestrequest,HttpServletResponseresponse){// 验证用户名密码...StringuserId="user-1001";StringaccessToken=jwtUtil.generateAccessToken(userId,30);// 30分钟StringrefreshToken=UUID.randomUUID().toString();// 存 redis(带过期时间)redisTemplate.opsForValue().set("refresh:"+refreshToken,userId,14,TimeUnit.DAYS);// 设置 HttpOnly CookieCookierefreshCookie=newCookie("rt",refreshToken);refreshCookie.setHttpOnly(true);refreshCookie.setSecure(true);// 生产必须 httpsrefreshCookie.setPath("/");refreshCookie.setMaxAge(14*24*60*60);refreshCookie.setAttribute("SameSite","Strict");response.addCookie(refreshCookie);returnResponseEntity.ok(newTokenVO(accessToken));}// 刷新 token 接口@PostMapping("/api/auth/refresh")publicResponseEntity<?>refresh(HttpServletRequestrequest){StringrefreshToken=null;Cookie[]cookies=request.getCookies();if(cookies!=null){refreshToken=Arrays.stream(cookies).filter(c->"rt".equals(c.getName())).map(Cookie::getValue).findFirst().orElse(null);}if(refreshToken==null){returnResponseEntity.status(401).body("无 refresh token");}StringuserId=redisTemplate.opsForValue().get("refresh:"+refreshToken);if(userId==null){returnResponseEntity.status(401).body("refresh token 已失效");}// 可选:滚动刷新(生成新 refresh token)StringnewRefreshToken=UUID.randomUUID().toString();redisTemplate.delete("refresh:"+refreshToken);redisTemplate.opsForValue().set("refresh:"+newRefreshToken,userId,14,TimeUnit.DAYS);// 返回新 access tokenreturnResponseEntity.ok(newTokenVO(jwtUtil.generateAccessToken(userId,30)));}

前端 Axios 拦截器(最经典写法)

// request 拦截器 - 自动加 tokenapi.interceptors.request.use(config=>{consttoken=localStorage.getItem('access_token')if(token){config.headers.Authorization=`Bearer${token}`}returnconfig})// response 拦截器 - 401 自动刷新api.interceptors.response.use(response=>response,asyncerror=>{constoriginalRequest=error.configif(error.response?.status===401&&!originalRequest._retry){originalRequest._retry=truetry{const{data}=awaitapi.post('/auth/refresh')// 自动携带 cookielocalStorage.setItem('access_token',data.accessToken)// 重试原请求originalRequest.headers.Authorization=`Bearer${data.accessToken}`returnapi(originalRequest)}catch(refreshErr){// 刷新失败 → 跳转登录localStorage.removeItem('access_token')window.location.href='/login'returnPromise.reject(refreshErr)}}returnPromise.reject(error)})

2. 文件上传下载(大文件、分片、断点续传)

三种主流方式对比(2026企业真实选型)

场景推荐方式最大文件断点续传复杂度代表技术栈
小文件(<50MB)普通 multipart50~100MB不支持★☆☆☆☆spring.servlet.multipart
中大文件(100MB~2GB)分片上传 + 秒传无上限支持★★★☆☆tus / Resumable.js / 前端分片
超大文件、企业级需求分片 + Redis 记录无上限完美支持★★★★☆minio + redis + 前端分片

推荐中型项目写法:分片上传 + 秒传(Redis 判断)

后端核心代码

@PostMapping("/upload/chunk")publicResponseEntity<?>uploadChunk(@RequestParam("file")MultipartFilechunk,@RequestParam("md5")StringfileMd5,@RequestParam("chunkIndex")intchunkIndex,@RequestParam("totalChunks")inttotalChunks){// 1. 秒传判断if(redisTemplate.hasKey("file:md5:"+fileMd5)){returnResponseEntity.ok("秒传成功");}// 2. 保存分片(临时目录 + chunkIndex 命名)StringchunkPath=uploadDir+"/"+fileMd5+"/"+chunkIndex;chunk.transferTo(newFile(chunkPath));// 3. 记录已上传分片redisTemplate.opsForSet().add("chunks:"+fileMd5,String.valueOf(chunkIndex));// 4. 判断是否全部上传完成LonguploadedCount=redisTemplate.opsForSet().size("chunks:"+fileMd5);if(uploadedCount==totalChunks){// 合并分片(异步任务或立即合并)mergeChunks(fileMd5,totalChunks);// 记录完成redisTemplate.opsForValue().set("file:md5:"+fileMd5,"completed",7,TimeUnit.DAYS);}returnResponseEntity.ok("分片上传成功");}

前端推荐库(2026年最流行):

  • resumable.js(最稳定)
  • uploader.js/tus-js-client(支持 tus 协议)
  • spark-md5(前端算 MD5 实现秒传)

3. 实时消息推送(WebSocket vs SSE 终极对比实战)

2026年真实选型表

需求场景首选技术双向通信浏览器兼容断线重连实现复杂度代表框架/库
聊天、IM、协作编辑WebSocket极好需手动★★★★☆Spring WebSocket + STOMP
服务器主动推送(通知、进度)SSE极好自动★★☆☆☆SseEmitter
海量用户单向广播SSE极好自动★★☆☆☆Redis Pub/Sub + SSE
低频、兼容性极致要求Long Polling最好自动★☆☆☆☆基本不用了

SSE 推荐写法(最简单、最高性价比)

@GetMapping(value="/progress/{taskId}",produces=MediaType.TEXT_EVENT_STREAM_VALUE)publicSseEmitterprogress(@PathVariableStringtaskId){SseEmitteremitter=newSseEmitter(Long.MAX_VALUE);// 长连接// 异步推送进度Executors.newSingleThreadExecutor().submit(()->{try{for(inti=0;i<=100;i+=10){emitter.send(SseEmitter.event().name("progress").data(newProgressVO(i,"处理中...")));Thread.sleep(1000);}emitter.send(SseEmitter.event().name("complete").data("任务完成"));emitter.complete();}catch(Exceptione){emitter.completeWithError(e);}});returnemitter;}

前端 SSE 写法

constsource=newEventSource(`/api/progress/${taskId}`);source.addEventListener('progress',e=>{constdata=JSON.parse(e.data)console.log(`进度:${data.percent}% -${data.message}`)})source.addEventListener('complete',e=>{console.log('任务完成!')source.close()})source.onerror=()=>{console.log('连接断开,浏览器会自动重连...')}

总结:三大核心交互的企业级优先级排序(2026)

  1. 必须掌握:前后端分离 Token 认证 + 无感刷新(几乎所有项目都有)
  2. 强烈建议:文件分片上传 + 秒传(中大型项目必备)
  3. 根据业务选:SSE(单向推送首选) > WebSocket(双向实时首选)

你目前项目里最常遇到哪个交互的痛点?
是 Token 刷新并发问题?大文件上传卡顿?还是实时推送掉线重连?

告诉我,我可以给你更针对性的代码优化或避坑方案~

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/3/25 12:30:51

YOLO26 接入实时视频

1. 创建环境conda create --name yolo_new python3.102. 安装 yolopip install -U ultralytics3. 编写获取视频代码1. 环境&#xff1a;MacOS iphone 摄像头2. 导入相应的头文件import cv2 from ultralytics import solutions2. 获取摄像头def open_iphone_camera_with_cv():&…

作者头像 李华
网站建设 2026/4/2 10:30:36

毕业论文答辩PPT的终极解决方案:ChatPPT如何让我从焦虑到从容应对

毕业论文答辩PPT的终极解决方案&#xff1a;ChatPPT如何让我从焦虑到从容应对 凌晨两点&#xff0c;宿舍灯光下&#xff0c;我盯着电脑屏幕上三万字的论文&#xff0c;手指在键盘上悬停了十分钟——如何把这篇密密麻麻的文字浓缩成20页PPT&#xff1f;这曾是每个毕业生的噩梦。…

作者头像 李华
网站建设 2026/3/23 2:39:40

从混沌到秩序:Lisp家族的演化与ANSI标准化的必然

从混沌到秩序&#xff1a;Lisp家族的演化与ANSI标准化的必然 1.1.1 Lisp的基因溯源&#xff1a;从λ演算到符号计算革命 Lisp的诞生可以追溯到1958年&#xff0c;由John McCarthy在MIT发明。其核心灵感来源于Alonzo Church的λ演算&#xff0c;这一数学模型构成了现代函数式编程…

作者头像 李华
网站建设 2026/3/30 20:17:20

基于STM32单片机共享无线充电锂电池充电宝系统设计DIY21-640

本系统由STM32F103C8T6单片机核心板、继电器控制、蓝牙控制及电源组成。1、手机连接蓝牙设备。2、设备上贴有二维码&#xff0c;通过手机可以扫描&#xff0c;如果二维码解析正确&#xff0c;可以控制打开共享设备的继电器。3、如果使用完需要关闭设备开关&#xff08;即继电器…

作者头像 李华