小编典典

Spring Data JPA:如何在不引用父级中的子级的情况下启用级联删除?

spring-boot

也许这是一个过于简单的问题,但是当我尝试删除用户实体时却出现了异常。

用户实体:

@Entity
@Table(name = "users")
public class User 
{
    @Transient
    private static final int SALT_LENGTH = 32;

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private int id;

    @NotNull
    private String firstName;

    @NotNull
    private String lastName;

    @Column(unique = true, length = 254)
    @NotNull
    private String email;

    // BCrypt outputs 60 character results.
    @Column(length = 60)
    private String hashedPassword;

    @NotNull
    private String salt;

    private boolean enabled;

    @CreationTimestamp
    @Temporal(TemporalType.TIMESTAMP)
    @Column(updatable = false)
    private Date createdDate;

我有一个实体类,该实体类使用外键引用用户。我想发生的是,当删除用户时PasswordResetToken,引用该用户的所有对象也会被删除。我怎样才能做到这一点?

@Entity
@Table(name = "password_reset_tokens")
public class PasswordResetToken 
{
    private static final int EXPIRATION_TIME = 1; // In minutes

    private static final int RESET_CODE_LENGTH = 10;

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private int id;

    private String token;

    @OneToOne(targetEntity = User.class, fetch = FetchType.EAGER)
    @JoinColumn(nullable = false, name = "userId")
    private User user;

    private Date expirationDate;

我正在将异常归结为 Cannot delete or update a parent row: a foreign key constraint fails (heroku_bc5bfe73a752182.password_reset_tokens, CONSTRAINTFKk3ndxg5xp6v7wd4gjyusp15gqFOREIGN KEY (user_id) REFERENCESusers(id))

我想避免PasswordResetToken在父实体中添加对的引用,因为User无需了解PasswordResetToken


阅读 481

收藏
2020-05-30

共1个答案

小编典典

如果不创建双向关系,则在JPA级别上是不可能的。您需要在User类中指定级联类型。User应该是关系的所有者,并且应该提供有关如何处理关系的信息PasswordResetToken

但是,如果您不能建立双向关系,我建议您直接在模式生成SQL脚本中设置关系。

如果您通过SQL脚本而不是通过JPA自动生成来创建架构(我相信所有严肃的项目都必须遵循这种模式),则可以在其中添加ON DELETE CASCADE约束。

它看起来像这样:

CREATE TABLE password_reset_tokens (
  -- columns declaration here
  user_id INT(11) NOT NULL,
  CONSTRAINT FK_PASSWORD_RESET_TOKEN_USER_ID
  FOREIGN KEY (user_id) REFERENCES users (id)
    ON DELETE CASCADE
);

这是有关如何在Spring Boot中使用数据库迁移工具的文档。这是有关如何从hibernate中生成模式脚本的信息(这将简化编写自己的脚本的过程)。

2020-05-30