Spring不将组合键实体映射到数据库

时间:2020-09-22 05:41:09

标签: spring spring-boot spring-mvc jpa spring-data-jpa

我有两个实体User和Focus,我必须为其存储一个等级,这样一个User才能具有多个Focus的排名,并且一个Focus可能有许多用户。为了解决这个问题,我遵循了https://www.baeldung.com/jpa-many-to-many第3部分,其中有以下2个类:

@Entity(name = "UserBaseRatings")
public class Rating {

    @EmbeddedId
    RatingKey id;

    @ManyToOne
    @MapsId("userID")
    @JoinColumn(name="userID")
    private User user;

    @ManyToOne
    @MapsId("focusID")
    @JoinColumn(name="focusID")
    private Focus focus;

    private BigDecimal baseRating;

    protected Rating(){}

    public Rating(User user, Focus focus, BigDecimal baseRating) {
        this.user = user;
        this.focus = focus;
        this.baseRating = baseRating;
    }

    public User getUser() {
        return user;
    }

    public void setUser(User user) {
        this.user = user;
    }

    public Focus getFocus() {
        return focus;
    }

    public void setFocus(Focus focus) {
        this.focus = focus;
    }

    public BigDecimal getBaseRating() {
        return baseRating;
    }

    public void setBaseRating(BigDecimal baseRating) {
        this.baseRating = baseRating;
    }
}

@Embeddable
public class RatingKey implements Serializable {

    @Column(name="userID")
    private Long userID;

    @Column(name="focusID")
    private Long focusID;

    protected RatingKey(){}

    public RatingKey(Long userID, Long focusID) {
        this.userID = userID;
        this.focusID = focusID;
    }

    public Long getUserID() {
        return userID;
    }

    public void setUserID(Long userID) {
        this.userID = userID;
    }

    public Long getFocusID() {
        return focusID;
    }

    public void setFocusID(Long focusID) {
        this.focusID = focusID;
    }

    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        RatingKey ratingKey = (RatingKey) o;
        return Objects.equals(userID, ratingKey.userID) &&
                Objects.equals(focusID, ratingKey.focusID);
    }

    @Override
    public int hashCode() {
        return Objects.hash(userID, focusID);
    }
}

UserBaseRatings的表的baseRating的默认值为1,这样,如果用户对当前焦点没有评级,则他们的评级应为1。如果用户未进行评级,则会遇到问题一个等级,则该等级的集合为空,而不是每个存在的焦点的等级为1。

用户:

public class User {

    @Id
    @GeneratedValue(strategy=GenerationType.IDENTITY)
    private Long userID;

    @Column(name = "userHashedPassword")
    private String password;

    @Column(name = "userName")
    private String userName;
    @Column(name = "userEmail")
    private String email;

    //probably being reset
    @Transient
    private List<String> groups = new LinkedList<>();

    @ManyToMany
    @JoinTable(name = "UserRoles",
        joinColumns = @JoinColumn(
                name = "userID"),
            inverseJoinColumns = @JoinColumn(
                    name = "roleID"))
    private Set<Role> roles = new HashSet<>();

    @OneToMany(mappedBy = "user")
    private Set<Rating> ratings;

    protected User(){}

    public User(String userHashedPassword, String userName, String email, Set<Role> roles){
        this.password = userHashedPassword;
        this.userName = userName;
        this.email = email;
        this.roles = roles;
    }

    public Long getUserId() {
        return userID;
    }

    public void setId(Long userID) {
        this.userID = userID;
    }

    public String getPassword(){
        return password;
    }

    public void setPassword(String password){
        this.password = password;
    }

    public String getUsername() {
        return userName;
    }

    public void setUsername(String name) {
        this.userName = name;
    }

    public String getEmail() {
        return email;
    }

    public void setEmail(String email) {
        this.email = email;
    }

    public Set<Role> getRoles() {
        return roles;
    }

    public void setRoles(Set<Role> roles) {
        this.roles = roles;
    }

    public Set<Rating> getRatings() {
        return ratings;
    }

    public void setRatings(Set<Rating> ratings) {
        this.ratings = ratings;
    }
}

重点:

@Entity
public class Focus {
    @Id
    @GeneratedValue(strategy=GenerationType.AUTO)
    private Long focusID;

    private String focusCategory;

    private String focusName;

    private String focusExplanation;

    @OneToMany(mappedBy = "focus")
    Set<Rating> ratings;

    //image

    protected Focus(){}

    public Focus(String focusCategory, String focusName, String focusExplanation, Set<Rating> ratings){
        this.focusCategory = focusCategory;
        this.focusName = focusName;
        this.focusExplanation = focusExplanation;
        this.ratings = ratings;
    }

    public Long getFocusId() {
        return focusID;
    }

    public void setFocusId(Long focusID) {
        this.focusID = focusID;
    }

    public String getFocusCategory() {
        return focusCategory;
    }

    public void setFocusCategory(String focusCategory) {
        this.focusCategory = focusCategory;
    }

    public String getFocusName() {
        return focusName;
    }

    public void setFocusName(String focusName) {
        this.focusName = focusName;
    }

    public String getFocusExplanation() {
        return focusExplanation;
    }

    public void setFocusExplanation(String focusExplanation) {
        this.focusExplanation = focusExplanation;
    }

    public Set<Rating> getRatings() {
        return ratings;
    }

    public void setRatings(Set<Rating> ratings) {
        this.ratings = ratings;
    }
}

我缺少什么,可以使用数据库中存在的每个焦点的默认值填充用户的评分?

编辑:将@PrePersist和@PreUpdate添加到评级实体:

@Entity(name = "UserBaseRatings")
public class Rating {

    @EmbeddedId
    RatingKey id;

    @ManyToOne
    @MapsId("userID")
    @JoinColumn(name="userID")
    private User user;

    @ManyToOne
    @MapsId("focusID")
    @JoinColumn(name="focusID")
    private Focus focus;

    private BigDecimal baseRating;

    protected Rating(){}

    public Rating(User user, Focus focus, BigDecimal baseRating) {
        this.user = user;
        this.focus = focus;
        this.baseRating = baseRating;
    }

    public User getUser() {
        return user;
    }

    public void setUser(User user) {
        this.user = user;
    }

    public Focus getFocus() {
        return focus;
    }

    public void setFocus(Focus focus) {
        this.focus = focus;
    }

    public BigDecimal getBaseRating() {
        return baseRating;
    }

    public void setBaseRating(BigDecimal baseRating) {
        this.baseRating = baseRating;
    }

    @PrePersist
    public void prePersist() {
        if(baseRating == null) {
            baseRating = BigDecimal.ONE;
        }
    }

    @PreUpdate
    public void preUpdate() {
        if(baseRating == null) {
            baseRating = BigDecimal.ONE;
        }
    }
}

spring.jpa.hibernate.ddl-auto=none更改为update也没什么区别。

1 个答案:

答案 0 :(得分:0)

如果使用ORM,则必须显式声明所有默认值。可以通过这种方式完成

@Entity(name = "UserBaseRatings")
public class Rating {

    // ...

    private BigDecimal baseRating = BigDecimal.ONE;

    // ...
}

或使用@PrePersist@PreUpdate

@Entity(name = "UserBaseRatings")
public class Rating {
    
    // ...
    
    private BigDecimal baseRating;
        
    // ...
    
    @PrePersist
    public void prePersist() {
        if(baseRating == null) {
           baseRating = BigDecimal.ONE;
        }
    }  
    
    @PreUpdate
    public void preUpdate() {
        if(baseRating == null) {
           baseRating = BigDecimal.ONE;
        }
    }    
}