0

0

Spring Boot 中实现用户数据访问权限控制的完整指南

碧海醫心

碧海醫心

发布时间:2026-01-15 10:42:14

|

119人浏览过

|

来源于php中文网

原创

Spring Boot 中实现用户数据访问权限控制的完整指南

本文介绍如何在 spring boot 应用中为 rest 接口添加细粒度安全控制,确保用户仅能访问自身数据(如 `/utente` 接口),防止 id 滥用导致的数据越权泄露。核心方案包括启用方法级安全、集成认证上下文、校验请求主体身份与资源归属一致性。

要真正实现“仅登录用户可查看自己的资料”,仅靠 @Secured 注解是不够的——它只能控制“谁有权限调用该方法”,但无法解决“用户 A 传入 ID=2 试图读取用户 B 的数据”这一典型越权问题(Insecure Direct Object Reference, IDOR)。你需要结合认证上下文感知 + 业务逻辑校验双重防护。

✅ 正确做法:三步加固

1. 启用方法级安全(基础前提)

在主配置类(如 SecurityConfig)中启用全局方法安全:

@Configuration
@EnableWebSecurity
@EnableMethodSecurity // 替代旧版 @EnableGlobalMethodSecurity
public class SecurityConfig {
    // ... 其他配置
}
⚠️ 注意:@EnableMethodSecurity 是 Spring Security 5.6+ 推荐方式,自动启用 @PreAuthorize、@PostAuthorize 等注解支持。

2. 使用 @PreAuthorize 校验调用者身份(声明式控制)

修改你的控制器方法,不接收外部传入的 idUtente,而是从认证上下文中提取当前登录用户的 ID:

@PostMapping("/utente")
public ResponseEntity getDatiProfiloUtente(
        @AuthenticationPrincipal CustomUserDetails userDetails) { // 假设你使用自定义 UserDetails

    Long currentUserId = userDetails.getId(); // 从认证对象获取真实用户 ID
    Paziente paziente = service.findPazienteById(currentUserId);

    if (paziente == null) {
        return ResponseEntity.status(HttpStatus.NOT_FOUND)
                .body(Map.of("error", "Profilo non trovato"));
    }

    Map map = new HashMap<>();
    map.put("nome", paziente.getNome());
    map.put("cognome", paziente.getCognome());
    map.put("email", paziente.getEmail());
    map.put("nTelefono", paziente.getNumeroTelefono());
    map.put("emailCaregiver", paziente.getEmailCaregiver());
    map.put("nomeCaregiver", paziente.getNomeCaregiver());
    map.put("cognomeCaregiver", paziente.getCognomeCaregiver());

    return ResponseEntity.ok(map);
}

✅ 优势:完全消除客户端传 ID 的风险;天然绑定会话身份。

3. (进阶)若必须支持“查他人”场景,用 @PostAuthorize 动态校验

例如管理员可查任意用户,但普通用户只能查自己:

DeepL
DeepL

DeepL是一款强大的在线AI翻译工具,可以翻译31种不同语言的文本,并可以处理PDF、Word、PowerPoint等文档文件

下载
@GetMapping("/utente/{id}")
@PostAuthorize("returnObject != null && " +
               "(hasRole('ADMIN') || #id == authentication.principal.id)")
public ResponseEntity getPazienteById(@PathVariable Long id) {
    return ResponseEntity.ok(service.findPazienteById(id));
}

此注解在方法执行后校验返回值和权限,确保即使方法返回了数据,也会被拦截非授权访问。

? 关键注意事项

  • 永远不要信任客户端传入的 ID:删除 @RequestBody Long idUtente 这种设计,这是 IDOR 漏洞根源。
  • 确保 UserDetails 实现包含用户唯一标识(如 id 字段),并在登录时正确注入 Authentication。
  • 前端未就绪 ≠ 安全可妥协:Postman 测试阶段更需严格模拟真实权限流,建议配合 JWT 或 Session 登录后再测试接口。
  • 补充传输层安全:生产环境务必启用 HTTPS,避免认证凭据(如 token)被窃听。

✅ 总结

真正的安全不是“加个注解就完事”,而是:

  1. 身份可信(通过 @AuthenticationPrincipal 获取真实用户)
  2. 数据归属校验(服务层确认 paziente.getUserId() == currentUser.getId())
  3. 权限分层控制(@PreAuthorize 控制入口,@PostAuthorize 控制结果)

这样,即使攻击者知道任意 ID,也无法绕过身份绑定逻辑,彻底杜绝越权访问风险。

相关专题

更多
spring框架介绍
spring框架介绍

本专题整合了spring框架相关内容,想了解更多详细内容,请阅读专题下面的文章。

102

2025.08.06

spring boot框架优点
spring boot框架优点

spring boot框架的优点有简化配置、快速开发、内嵌服务器、微服务支持、自动化测试和生态系统支持。本专题为大家提供spring boot相关的文章、下载、课程内容,供大家免费下载体验。

135

2023.09.05

spring框架有哪些
spring框架有哪些

spring框架有Spring Core、Spring MVC、Spring Data、Spring Security、Spring AOP和Spring Boot。详细介绍:1、Spring Core,通过将对象的创建和依赖关系的管理交给容器来实现,从而降低了组件之间的耦合度;2、Spring MVC,提供基于模型-视图-控制器的架构,用于开发灵活和可扩展的Web应用程序等。

389

2023.10.12

Java Spring Boot开发
Java Spring Boot开发

本专题围绕 Java 主流开发框架 Spring Boot 展开,系统讲解依赖注入、配置管理、数据访问、RESTful API、微服务架构与安全认证等核心知识,并通过电商平台、博客系统与企业管理系统等项目实战,帮助学员掌握使用 Spring Boot 快速开发高效、稳定的企业级应用。

68

2025.08.19

Java Spring Boot 4更新教程_Java Spring Boot 4有哪些新特性
Java Spring Boot 4更新教程_Java Spring Boot 4有哪些新特性

Spring Boot 是一个基于 Spring 框架的 Java 开发框架,它通过 约定优于配置的原则,大幅简化了 Spring 应用的初始搭建、配置和开发过程,让开发者可以快速构建独立的、生产级别的 Spring 应用,无需繁琐的样板配置,通常集成嵌入式服务器(如 Tomcat),提供“开箱即用”的体验,是构建微服务和 Web 应用的流行工具。

32

2025.12.22

Java Spring Boot 微服务实战
Java Spring Boot 微服务实战

本专题深入讲解 Java Spring Boot 在微服务架构中的应用,内容涵盖服务注册与发现、REST API开发、配置中心、负载均衡、熔断与限流、日志与监控。通过实际项目案例(如电商订单系统),帮助开发者掌握 从单体应用迁移到高可用微服务系统的完整流程与实战能力。

114

2025.12.24

软件测试常用工具
软件测试常用工具

软件测试常用工具有Selenium、JUnit、Appium、JMeter、LoadRunner、Postman、TestNG、LoadUI、SoapUI、Cucumber和Robot Framework等等。测试人员可以根据具体的测试需求和技术栈选择适合的工具,提高测试效率和准确性 。

436

2023.10.13

session失效的原因
session失效的原因

session失效的原因有会话超时、会话数量限制、会话完整性检查、服务器重启、浏览器或设备问题等等。详细介绍:1、会话超时:服务器为Session设置了一个默认的超时时间,当用户在一段时间内没有与服务器交互时,Session将自动失效;2、会话数量限制:服务器为每个用户的Session数量设置了一个限制,当用户创建的Session数量超过这个限制时,最新的会覆盖最早的等等。

307

2023.10.17

Golang gRPC 服务开发与Protobuf实战
Golang gRPC 服务开发与Protobuf实战

本专题系统讲解 Golang 在 gRPC 服务开发中的完整实践,涵盖 Protobuf 定义与代码生成、gRPC 服务端与客户端实现、流式 RPC(Unary/Server/Client/Bidirectional)、错误处理、拦截器、中间件以及与 HTTP/REST 的对接方案。通过实际案例,帮助学习者掌握 使用 Go 构建高性能、强类型、可扩展的 RPC 服务体系,适用于微服务与内部系统通信场景。

8

2026.01.15

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Node.js 教程
Node.js 教程

共57课时 | 8.6万人学习

CSS3 教程
CSS3 教程

共18课时 | 4.5万人学习

Vue 教程
Vue 教程

共42课时 | 6.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号