Spring Boot中医生-患者关系的高效数据模型与安全实践

Spring Boot中医生-患者关系的高效数据模型与安全实践

本文探讨了在Spring Boot应用中构建复杂用户关系(如医生与患者)的数据模型和安全集成方案。通过采用共享的用户基类与特定角色实体的混合模式,我们能够清晰地分离通用用户属性与角色特有数据,有效管理多对多关系,并基于用户角色实现灵活的权限控制,同时避免了数据冗余和空值问题,提供了一种健壮且可扩展的解决方案。

在开发医疗管理系统时,如何高效地建模医生与患者之间的复杂关系,并在此基础上集成灵活的用户认证与授权机制,是常见的挑战。传统的解决方案往往面临两个主要困境:一是为医生和患者创建完全独立的实体,这导致在用户登录和权限管理时逻辑复杂;二是将所有属性集中到一个用户实体中,通过角色区分,但这可能导致大量空字段和不必要的业务逻辑分支。本教程将介绍一种更优的数据模型设计,并演示如何与spring security集成,以构建一个既清晰又功能强大的医生-患者关系管理系统。

核心数据模型设计

为了解决上述问题,我们采用一种混合模型:一个通用的User实体来存储所有用户的通用信息(如登录凭据、姓名),以及独立的Doctor和Patient实体来存储各自特有的属性和关系。这种设计通过@OneToOne和@MapsId注解将特定角色实体与User实体关联起来,确保了数据的一致性和模型的灵活性。

1. 用户实体 (User)

User实体作为所有用户的基类,包含通用的认证信息和基本身份属性。

import javax.persistence.*;import lombok.Getter;import lombok.Setter;@Entity@Table(name = "app_users") // 避免与数据库保留字冲突@Getter@Setterpublic class User {    @Id    @GeneratedValue(strategy = GenerationType.IDENTITY)    private Long id;    @Column(unique = true, nullable = false)    private String username; // 用户名,用于登录    @Column(nullable = false)    private String password; // 密码,应加密存储    private String name;    private String surname;    @Enumerated(EnumType.STRING)    @Column(nullable = false)    private UserType userType; // 用户类型,如医生、患者}
// UserType 枚举定义public enum UserType {    DOCTOR,    PATIENT,    ADMIN // 可根据需求扩展}

2. 医生实体 (Doctor)

Doctor实体存储医生特有的信息,并通过@MapsId与User实体建立一对一关联。医生与患者之间存在多对多关系。

import javax.persistence.*;import java.util.HashSet;import java.util.Set;import lombok.Getter;import lombok.Setter;@Entity@Getter@Setterpublic class Doctor {    @Id    private Long id; // 与User的ID保持一致    @OneToOne    @MapsId // 表示Doctor的主键也是其关联User的外键    @JoinColumn(name = "id") // 外键列名    private User user;    @ManyToMany(cascade = {CascadeType.PERSIST, CascadeType.MERGE})    @JoinTable(        name = "doctor_patient", // 中间表名        joinColumns = @JoinColumn(name = "doctor_id"), // 医生在中间表中的列名        inverseJoinColumns = @JoinColumn(name = "patient_id") // 患者在中间表中的列名    )    private Set patients = new HashSet();    // 辅助方法,用于维护双向关系    public void addPatient(Patient patient) {        this.patients.add(patient);        patient.getDoctors().add(this);    }    public void removePatient(Patient patient) {        this.patients.remove(patient);        patient.getDoctors().remove(this);    }}

3. 患者实体 (Patient)

Patient实体存储患者特有的信息,同样通过@MapsId与User实体关联。患者可以拥有多位医生,并记录所服用的药物信息。

import javax.persistence.*;import java.util.HashSet;import java.util.Set;import lombok.Getter;import lombok.Setter;@Entity@Getter@Setterpublic class Patient {    @Id    private Long id; // 与User的ID保持一致    @OneToOne    @MapsId    @JoinColumn(name = "id")    private User user;    @ManyToMany(mappedBy = "patients", fetch = FetchType.LAZY) // 由Doctor实体中的patients字段映射    private Set doctors = new HashSet();    @ManyToMany(cascade = {CascadeType.PERSIST, CascadeType.MERGE})    @JoinTable(        name = "patient_medicine",        joinColumns = @JoinColumn(name = "patient_id"),        inverseJoinColumns = @JoinColumn(name = "medicine_id")    )    private Set medicines = new HashSet();    // 辅助方法,用于维护双向关系    public void addMedicine(Medicine medicine) {        this.medicines.add(medicine);        medicine.getPatients().add(this);    }    public void removeMedicine(Medicine medicine) {        this.medicines.remove(medicine);        medicine.getPatients().remove(this);    }}

4. 药物实体 (Medicine)

Medicine实体存储药物的基本信息,与Patient实体存在多对多关系。

import javax.persistence.*;import java.util.HashSet;import java.util.Set;import lombok.Getter;import lombok.Setter;@Entity@Getter@Setterpublic class Medicine {    @Id    @GeneratedValue(strategy = GenerationType.IDENTITY)    private Long id;    private String name;    @ManyToMany(mappedBy = "medicines", fetch = FetchType.LAZY) // 由Patient实体中的medicines字段映射    private Set patients = new HashSet();}

Spring Security 集成

这种数据模型为Spring Security的集成提供了清晰的路径。User实体将作为Spring Security进行认证的基石,而UserType则用于授权和区分不同角色的业务逻辑。

1. 实现 UserDetailsService

UserDetailsService是Spring Security的核心接口,用于加载用户详情。我们将通过username从User实体中加载用户信息,并根据UserType赋予相应的角色权限。

import org.springframework.security.core.userdetails.UserDetails;import org.springframework.security.core.userdetails.UserDetailsService;import org.springframework.security.core.userdetails.UsernameNotFoundException;import org.springframework.security.core.authority.SimpleGrantedAuthority;import org.springframework.stereotype.Service;import java.util.Collections;import java.util.List;import java.util.stream.Collectors;@Servicepublic class CustomUserDetailsService implements UserDetailsService {    private final UserRepository userRepository; // 假设您有一个UserRepository接口    public CustomUserDetailsService(UserRepository userRepository) {        this.userRepository = userRepository;    }    @Override    public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException {        User user = userRepository.findByUsername(username)                .orElseThrow(() -> new UsernameNotFoundException("用户未找到: " + username));        // 根据UserType构建Spring Security的权限列表        List authorities = Collections.singletonList(            new SimpleGrantedAuthority("ROLE_" + user.getUserType().name())        );        // 返回Spring Security的User对象        return new org.springframework.security.core.userdetails.User(                user.getUsername(),                user.getPassword(), // 密码应为加密后的形式                authorities        );    }}

注意: UserRepository需要您自行定义,例如:public interface UserRepository extends JpaRepository { Optional findByUsername(String username); }。同时,在实际应用中,User实体中的password字段应存储加密后的密码,并在注册用户时使用PasswordEncoder进行编码。

2. 授权管理

一旦用户通过认证,其UserType(通过ROLE_UserType形式的权限)就可以用于授权。您可以在控制器或服务层根据当前用户的角色来执行不同的业务逻辑,或者使用Spring Security的注解进行方法级别的权限控制。

import org.springframework.security.access.prepost.PreAuthorize;import org.springframework.security.core.Authentication;import org.springframework.security.core.context.SecurityContextHolder;import org.springframework.stereotype.Service;import java.util.Optional;@Servicepublic class HealthCareService {    private final UserRepository userRepository;    private final DoctorRepository doctorRepository; // 假设有DoctorRepository    private final PatientRepository patientRepository; // 假设有PatientRepository    public HealthCareService(UserRepository userRepository, DoctorRepository doctorRepository, PatientRepository patientRepository) {        this.userRepository = userRepository;        this.doctorRepository = doctorRepository;        this.patientRepository = patientRepository;    }    // 医生特有操作:查看其所有患者    @PreAuthorize

以上就是Spring Boot中医生-患者关系的高效数据模型与安全实践的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/107911.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月22日 17:55:25
下一篇 2025年11月22日 18:12:42

相关推荐

  • 使用自定义整型类型及其范围(Go语言)

    本文旨在阐述在Go语言中如何使用自定义整型类型,并解释了为什么在循环中使用 range 时需要显式类型转换。文章将深入探讨Go语言的类型系统,并提供代码示例来说明类型转换的必要性,以及如何在实际开发中正确地使用自定义整型类型。 在Go语言中,我们可以使用 type 关键字创建自定义类型,这在很多情况…

    好文分享 2025年12月15日
    000
  • Go App Engine 本地开发服务器启动:解决找不到Go文件异常

    本文针对Go App Engine示例应用在本地开发服务器启动时,因路径配置不当导致“找不到Go文件”的异常,提供了详细的解决方案。核心在于正确指定 dev_appserver.py 命令的应用目录,确保其能定位到包含 app.yaml 和 Go 源码的路径,从而避免运行时错误并成功启动应用。 理解…

    2025年12月15日
    000
  • Golang使用errors.Unwrap获取原始错误

    答案:errors.Unwrap用于获取被包装的底层错误,它通过调用错误的Unwrap方法剥离一层封装,适用于解析错误链。结合fmt.Errorf的%w动词,可构建支持解包的错误链。与errors.Is(判断错误值)和errors.As(判断错误类型)相比,Unwrap仅解包一层,是后两者的底层基础…

    2025年12月15日
    000
  • Golang工厂模式结合配置文件创建对象

    答案:将工厂模式与配置文件结合可在不修改代码情况下动态创建对象,提升系统解耦性、可配置性、可维护性与扩展性,支持运行时灵活调整对象类型和参数,适用于多环境部署与复杂初始化场景。 在Go语言中,将工厂模式与配置文件结合起来创建对象,说白了,就是为了让你的系统变得更“活”。它允许你在不修改、不重新编译代…

    2025年12月15日
    000
  • Golang布尔类型使用与逻辑运算实例

    Go语言中布尔类型bool仅取true或false,支持&&、||、!运算符并具有短路特性,严格类型安全避免隐式转换。通过if-else、for、switch等控制结构实现逻辑判断,合理使用卫语句、枚举和函数封装可提升代码可读性与健壮性。 Go语言中的布尔类型( bool )是处理逻…

    2025年12月15日
    000
  • Golang反射调用函数及方法完整示例

    Go语言反射可通过reflect.Value.Call动态调用函数和方法,示例包括调用add函数和Calculator的方法,支持多返回值与错误处理,需注意可访问性、参数匹配及性能开销。 Go语言的反射机制允许程序在运行时动态调用函数和方法,这对于实现通用库、框架(如序列化、依赖注入)非常有用。下面…

    2025年12月15日
    000
  • 去除 []byte 中的 C 风格注释

    本文介绍了如何使用 Go 语言去除 byte 数组中的 C 风格注释(包括单行 // 和多行 /* */ 注释)。通过使用正则表达式,我们可以有效地从 JSON 文件或其他文本数据中移除这些注释,使其符合 JSON 规范,从而能够使用 json.Unmarshal 等函数进行解析。 JSON(Jav…

    2025年12月15日
    000
  • Golang使用filepath处理路径操作技巧

    filepath包提供路径处理函数,如Clean清理冗余、Join安全拼接、Abs获取绝对路径、Walk遍历目录,结合os.Stat判断路径是否存在。 Golang的 filepath 包,说白了,就是让你在Go程序里优雅地处理文件路径的各种问题。它提供了一系列函数,帮你规范化路径、拼接路径、获取绝…

    2025年12月15日
    000
  • GolangRPC错误处理与状态码设计技巧

    设计Go RPC服务时需统一错误结构,使用结构化RPCError包含Code、Message和Details;映射gRPC标准状态码如InvalidArgument、NotFound;分层管理错误码,按1xx、2xx、3xx划分客户端、服务端、第三方错误;返回客户端信息应简洁友好,避免暴露技术细节,…

    2025年12月15日
    000
  • GolangWeb服务器负载均衡与性能提升

    通过反向代理实现负载均衡,部署多实例并优化Go服务性能,结合缓存与异步处理,提升系统吞吐量和稳定性。 构建高并发的 Web 服务时,Golang 因其轻量级协程和高效网络处理能力成为理想选择。但单机服务总有性能瓶颈,面对大量请求时,必须通过负载均衡和系统优化来提升整体吞吐能力和稳定性。以下是实现 G…

    2025年12月15日
    000
  • Go App Engine 本地开发:示例项目运行异常排查与解决方案

    本文旨在解决Go App Engine本地开发环境中运行示例项目时常见的“找不到Go文件”异常。核心问题在于dev_appserver.py脚本对应用目录结构的预期与实际示例项目结构不符。教程将详细解释错误原因,并提供通过指定正确应用路径来成功启动Go App Engine示例项目的解决方案,确保开…

    2025年12月15日
    000
  • Golang动态生成对象并赋值技巧

    答案:Golang中通过reflect包实现动态生成对象并赋值,利用reflect.New创建实例,FieldByName查找字段,SetInt、SetString等方法赋值,仅限可导出字段(首字母大写),且需通过Elem()获取可设置的Value;常用于通用数据解析、插件系统、ORM等场景,结合接…

    2025年12月15日
    000
  • GolangRPC多服务调用链跟踪实践

    通过统一TraceID透传、OpenTelemetry自动埋点、日志关联及合理采样策略,实现Golang微服务RPC调用链跟踪,提升跨服务问题排查效率。 在微服务架构中,一次请求往往会跨越多个服务,Golang 的 RPC 调用链路复杂时,排查问题变得困难。为了快速定位延迟、异常或性能瓶颈,需要实现…

    2025年12月15日
    000
  • Golanggoroutine泄漏检测与防护方法

    Go中goroutine泄漏主因包括channel阻塞、未关闭channel、无限循环无退出及子goroutine未随父退出;2. 通过pprof监控goroutine数量可检测泄漏;3. 使用context控制生命周期,如WithCancel发送取消信号,确保goroutine及时退出。 Go语言…

    2025年12月15日
    000
  • Golang云原生环境下日志聚合与分析实践

    答案:在Golang云原生环境中,实现高效可观测的结构化日志需选用zap等高性能日志库,结合context传递Trace ID等上下文信息,输出JSON格式日志;通过Fluent Bit或Fluentd收集日志,送至Loki或Elasticsearch存储;利用Grafana或Kibana进行查询分…

    2025年12月15日
    000
  • 解决 App Engine Go 示例抛出异常的问题

    在使用 Google App Engine 运行 Go 示例时,可能会遇到 “no .go files in %s” 的异常。本文旨在提供一个清晰的解决方案,帮助开发者正确运行 App Engine Go 示例,避免常见错误,并理解问题背后的原因。通过简单的命令行操作,您可以…

    2025年12月15日
    000
  • Golang文件IO性能优化与缓冲使用技巧

    Golang中文件IO性能优化的核心是减少系统调用和合理利用缓冲,主要通过bufio包实现。使用bufio.Reader和bufio.Writer可将多次小数据读写聚合成批量操作,显著降低用户态与内核态切换开销。例如,写入10万行文本时,无缓冲需数万次系统调用,而带缓冲可能仅需几次,性能差距巨大。可…

    2025年12月15日
    000
  • Golang文件权限与属性设置技巧

    Golang通过os.Chmod和os.Chown函数操作文件权限与所有者,结合os.FileMode使用八进制权限码(如0644、0755),需注意程序运行用户权限及umask影响;创建文件时应显式指定最小必要权限,特殊权限位(SUID、SGID、Sticky Bit)可通过按位或设置,但需谨慎评…

    2025年12月15日
    000
  • Golang错误处理链式调用与包装方法

    Golang错误包装通过%w构建可追溯的错误链,解决上下文丢失、调试困难等问题。使用fmt.Errorf(“%w”)在各逻辑层添加上下文,保留底层错误;errors.Is检查特定错误类型,errors.As提取自定义错误信息,实现精准错误判断与处理。最佳实践包括:在模块边界包…

    2025年12月15日
    000
  • Golang的errors.Is函数如何检查错误链中是否存在特定错误

    errors.Is用于判断错误链中是否存在指定错误,它通过递归遍历错误链实现深层比较,而直接比较仅判断错误实例是否相同;自定义错误可通过实现Is方法支持errors.Is;errors.As则用于判断并提取特定类型的错误;错误链过长理论上影响性能但实际可忽略;在测试中使用errors.Is可更可靠地…

    2025年12月15日
    000

发表回复

登录后才能评论
关注微信