簡體   English   中英

JPA外鍵約束違規無法插入Null

[英]JPA Foreign Key constraint violation cannot insert Null

我有一個簡單的User和Roles實體,具有一對多映射。 一個具有多個角色的用戶。 我嘗試保存用戶實體時收到錯誤。 我在調用persist之前設置了Role對象User。

以下是實體的快照。

用戶:

package com.petpe.ejbadmin.entity;

import java.io.Serializable;
import java.util.Date;
import java.util.Set;

import javax.annotation.Generated;
import javax.persistence.Basic;
import javax.persistence.CascadeType;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.FetchType;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.NamedQueries;
import javax.persistence.NamedQuery;
import javax.persistence.OneToMany;
import javax.persistence.Table;
import javax.persistence.Temporal;
import javax.persistence.TemporalType;
import javax.validation.constraints.NotNull;
import javax.validation.constraints.Size;
import javax.xml.bind.annotation.XmlRootElement;
import javax.xml.bind.annotation.XmlTransient;


@Entity
@Table(name = "users")
@XmlRootElement
@NamedQueries({
    @NamedQuery(name = "User.findAll", query = "SELECT u FROM User u"),
    @NamedQuery(name = "User.findByUsernameandPassword", query = "SELECT u FROM User u WHERE u.username = :username AND u.password = :password"),
    @NamedQuery(name = "User.findByUserfirstname", query = "SELECT u FROM User u WHERE u.userfirstname = :userfirstname"),
    @NamedQuery(name = "User.findByUserlastname", query = "SELECT u FROM User u WHERE u.userlastname = :userlastname"),
    @NamedQuery(name = "User.findByPhonenumber", query = "SELECT u FROM User u WHERE u.phonenumber = :phonenumber"),
    @NamedQuery(name = "User.findByEmail", query = "SELECT u FROM User u WHERE u.email = :email"),
    @NamedQuery(name = "User.findByErpid", query = "SELECT u FROM User u WHERE u.erpid = :erpid"),
    @NamedQuery(name = "User.findByBuyerid", query = "SELECT u FROM User u WHERE u.buyerid = :buyerid"),
    @NamedQuery(name = "User.findByCreatedDate", query = "SELECT u FROM User u WHERE u.createdDate = :createdDate"),
    @NamedQuery(name=  "User.findByfirstletter",query="SELECT u FROM User u join fetch u.RoleSet ur WHERE LOWER(u.userfirstname) LIKE :firstname"),
    @NamedQuery(name=  "User.findallusers",query="From User u join fetch u.RoleSet ur")})
public class User implements Serializable {
    private static final long serialVersionUID = 1L;
    @Id
    @GeneratedValue(strategy=GenerationType.IDENTITY)
    @Basic(optional = true)
    @Column(name = "USER_ID")
    private Integer userId;
    @Size(max = 120)
    @Column(name = "USERNAME")
    private String username;
    @Size(max = 60)
    @Column(name = "USERFIRSTNAME")
    private String userfirstname;
    @Size(max = 60)
    @Column(name = "USERLASTNAME")
    private String userlastname;
    @Size(max = 50)
    @Column(name = "PASSWORD")
    private String password;
    @Size(max = 20)
    @Column(name = "PHONENUMBER")
    private String phonenumber;
    // @Pattern(regexp="[a-z0-9!#$%&'*+/=?^_`{|}~-]+(?:\\.[a-z0-9!#$%&'*+/=?^_`{|}~-]+)*@(?:[a-z0-9](?:[a-z0-9-]*[a-z0-9])?\\.)+[a-z0-9](?:[a-z0-9-]*[a-z0-9])?", message="Invalid email")//if the field contains email address consider using this annotation to enforce field validation
    @Size(max = 50)
    @Column(name = "EMAIL")
    private String email;
    @Size(max = 250)
    @Column(name = "ERPID")
    private String erpid;
    @Size(max = 250)
    @Column(name = "BUYERID")
    private String buyerid;
    @Basic(optional = false)
    @NotNull
    @Column(name = "CREATED_DATE")
    @Temporal(TemporalType.DATE)
    private Date createdDate;
    @OneToMany(cascade = CascadeType.ALL, mappedBy = "user", fetch = FetchType.LAZY)
    private Set<Role> RoleSet;

    public User() {
    }

    public User(Integer userId) {
        this.userId = userId;
    }

    public User(Integer userId, Date createdDate) {
        this.userId = userId;
        this.createdDate = createdDate;
    }

    public Integer getUserId() {
        return userId;
    }

    public void setUserId(Integer userId) {
        this.userId = userId;
    }

    public String getUsername() {
        return username;
    }

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

    public String getUserfirstname() {
        return userfirstname;
    }

    public void setUserfirstname(String userfirstname) {
        this.userfirstname = userfirstname;
    }

    public String getUserlastname() {
        return userlastname;
    }

    public void setUserlastname(String userlastname) {
        this.userlastname = userlastname;
    }

    public String getPassword() {
        return password;
    }

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

    public String getPhonenumber() {
        return phonenumber;
    }

    public void setPhonenumber(String phonenumber) {
        this.phonenumber = phonenumber;
    }

    public String getEmail() {
        return email;
    }

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

    public String getErpid() {
        return erpid;
    }

    public void setErpid(String erpid) {
        this.erpid = erpid;
    }

    public String getBuyerid() {
        return buyerid;
    }

    public void setBuyerid(String buyerid) {
        this.buyerid = buyerid;
    }

    public Date getCreatedDate() {
        return createdDate;
    }

    public void setCreatedDate(Date createdDate) {
        this.createdDate = createdDate;
    }

    @XmlTransient
    public Set<Role> getRoleSet() {
        return RoleSet;
    }

    public void setRoleSet(Set<Role> RoleSet) {
        this.RoleSet = RoleSet;
    }

    @Override
    public int hashCode() {
        int hash = 0;
        hash += (userId != null ? userId.hashCode() : 0);
        return hash;
    }

    @Override
    public boolean equals(Object object) {
        // TODO: Warning - this method won't work in the case the id fields are not set
        if (!(object instanceof User)) {
            return false;
        }
        User other = (User) object;
        if ((this.userId == null && other.userId != null) || (this.userId != null && !this.userId.equals(other.userId))) {
            return false;
        }
        return true;
    }

    @Override
    public String toString() {
        return "[ userId=" + userId + " ]";
    }

}

角色:

/ * *要更改此許可證標題,請在“項目屬性”中選擇“許可證標題”。 *要更改此模板文件,請選擇“工具”| 模板*並在編輯器中打開模板。 * /

package com.petpe.ejbadmin.entity;

import java.io.Serializable;

import javax.persistence.Basic;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.FetchType;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.JoinColumn;
import javax.persistence.ManyToOne;
import javax.persistence.NamedQueries;
import javax.persistence.NamedQuery;
import javax.persistence.Table;
import javax.validation.constraints.NotNull;
import javax.validation.constraints.Size;
import javax.xml.bind.annotation.XmlRootElement;


@Entity
@Table(name = "user_roles")
@XmlRootElement
public class Role implements Serializable {
    private static final long serialVersionUID = 1L;
    @Id
    @GeneratedValue(strategy=GenerationType.IDENTITY)
    @Basic(optional = false)
    @Column(name = "ROLEID")
    private Integer roleid;
    @Basic(optional = false)
    @NotNull
    @Size(min = 1, max = 30)
    @Column(name = "ROLENAME")
    private String rolename;
    @JoinColumn(name = "USER_ID", referencedColumnName = "USER_ID")
    @ManyToOne( fetch = FetchType.LAZY)
    private User user;

    public Role() {
    }

    public Role(String rolename) {
        this.rolename = rolename;
    }

    public Role(Integer roleid, String rolename) {
        this.roleid = roleid;
        this.rolename = rolename;
    }

    public User getUser() {
        return user;
    }

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

    public Integer getRoleid() {
        return roleid;
    }

    public void setRoleid(Integer roleid) {
        this.roleid = roleid;
    }

    public String getRolename() {
        return rolename;
    }

    public void setRolename(String rolename) {
        this.rolename = rolename;
    }



    @Override
    public int hashCode() {
        int hash = 0;
        hash += (roleid != null ? roleid.hashCode() : 0);
        return hash;
    }

    @Override
    public boolean equals(Object object) {
        // TODO: Warning - this method won't work in the case the id fields are not set
        if (!(object instanceof Role)) {
            return false;
        }
        Role other = (Role) object;
        if ((this.roleid == null && other.roleid != null) || (this.roleid != null && !this.roleid.equals(other.roleid))) {
            return false;
        }
        return true;
    }

    @Override
    public String toString() {
        return "[ roleid=" + roleid + " ]";
    }

}

錯誤:

1.Cannot insert the value NULL into column 'USER_ID', table 'pedb.pedb.user_roles'; column does not allow nulls. INSERT fails.
2.could not insert: [com.petpe.ejbadmin.entity.Role]
3.org.hibernate.exception.ConstraintViolationException: could not insert: [com.petpe.ejbadmin.entity.Role]
4.javax.persistence.PersistenceException: org.hibernate.exception.ConstraintViolationException: could not insert: [com.petpe.ejbadmin.entity.Role]

設置和保留代碼:

public boolean AddorModifyUser(UserAdminDTO udato) {

        User usr=this.populateUserobj(udato);
        pet_em.persist(usr);

        return true;
    }

public User populateUserobj(UserAdminDTO udto)
    {
        User usr=new User();

        usr.setBuyerid(udto.getbuyerid());
        usr.setCreatedDate(new Date());
        usr.setEmail(udto.getEmail());
        usr.setErpid(udto.geterpid());
        usr.setPassword(udto.getPassword());
        usr.setPhonenumber(udto.getPhoneNumber());
        usr.setUserfirstname(udto.getFirstName());
        usr.setUserlastname(udto.getLastName());
        usr.setUsername(udto.getUserName());
        for(Role r: udto.getRole())
        {
            Set<Role> RoleSet=new HashSet<>();
            RoleSet.add(r);     
            usr.setRoleSet(RoleSet);
        }
        return usr;
    }

表格布局:

在此輸入圖像描述

因此,對於roles表中的USER_ID您有一個NOT NULL約束。 當持久化(即使具有級聯關系)時,外鍵(或實體關系)將不會自動設置自身。

最有可能在創建Role ,您沒有設置User ,因此在持久化時,提供程序不知道USER_ID是什么,因此嘗試在db中將其設置為null,這違反了db模式。

您需要做的就是為每個Role設置User

Set<Role> RoleSet=new HashSet<>();
for(Role r: udto.getRole())
{
    r.setUser(usr);   // <------
    RoleSet.add(r);     
}
usr.setRoleSet(RoleSet);

另請注意, Set是在循環外創建的

與...但是...等等...

你還沒有走出困境。 以上將解決錯誤,但你還有另一個問題。 它很微妙,肯定會導致很難找到錯誤。

您正在生成Role ID。

@Id
@GeneratedValue(strategy=GenerationType.IDENTITY)
private Integer roleid;

所以在創建角色時不要設置它

Set<Role> roles = new HashSet<>();
Role userRole = new Role("USER");
userRole.setUser(user);
roles.add(userRole);
Role coolRole = new Role("COOL_GUY");
coolRole.setUser(user);
roles.add(coolRole);
user.setRoleSet(roles);

看起來不是問題,但bug在hashCode

@Override
public int hashCode() {
    int hash = 0;
    hash += (roleid != null ? roleid.hashCode() : 0);
    return hash;
}

hashCode只包含roleId 但是你沒有設置roleId因為它應該被生成。 所以,當你添加的角色的Set ,它們都具有相同hashCode ,所以你最終只能添加一個RoleSet ,如在對象Set不能有相同hashCode

要解決此問題,您只需更改hashCode以包含rolename

在測試時,我只是讓Netbeans生成hashCodeequals 這是結果。

@Override
public int hashCode() {
    int hash = 7;
    hash = 31 * hash + Objects.hashCode(this.roleid);
    hash = 31 * hash + Objects.hashCode(this.rolename);
    return hash;
}

@Override
public boolean equals(Object obj) {
    if (obj == null) {
        return false;
    }
    if (getClass() != obj.getClass()) {
        return false;
    }
    final Role other = (Role) obj;
    if (!Objects.equals(this.roleid, other.roleid)) {
        return false;
    }
    if (!Objects.equals(this.rolename, other.rolename)) {
        return false;
    }
    return true;
}

那應該修復bug。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM