2011-11-14 77 views
6

我有数据库revisionsPagu休眠3复合键一个与GeneratedValue

Pagu模型这种双表格,我必须复合键:

  • ID INT(由数据库自动生成的)
  • revision_id(foreign_key to revision)表

如何在Hibernate 3上实现这个?

这就是我想出了

@Entity 
@Table(name="pagu" 
    ,schema="dbo" 
    ,catalog="dbname" 
) 
@IdClass(PaguId.class) 
public class Pagu implements java.io.Serializable { 

private int id; 
private int revisiId; 
private Entitas entitas; 
private Revisi revisi; 
... 

@Id 
@GeneratedValue 
@Column(name="id", unique=true, nullable=false) 
public int getId() { 
    return this.id; 
} 

public void setId(int id) { 
    this.id = id; 
} 

@Id 
@Column(name="revisi_id", unique=true, nullable=false) 
public int getRevisiId() { 
    return this.revisiId; 
} 

public void setRevisiId(int id) { 
    this.id = id; 
} 

这是我PaguId类

@Embeddable 
public class PaguId implements java.io.Serializable { 


    private int id; 
    private int revisiId; 

    public PaguId() { 
    } 

    public PaguId(int id, int revisiId) { 
     this.id = id; 
     this.revisiId = revisiId; 
    } 

    @Column(name="id", nullable=false) 
    public int getId() { 
     return this.id; 
    } 

    public void setId(int id) { 
     this.id = id; 
    } 

    @Column(name="revisi_id", nullable=false) 
    public int getRevisiId() { 
     return this.revisiId; 
    } 

    public void setRevisiId(int revisiId) { 
     this.revisiId = revisiId; 
    } 


    public boolean equals(Object other) { 
     if ((this == other)) return true; 
     if ((other == null)) return false; 
     if (!(other instanceof PaguId)) return false; 
     PaguId castOther = (PaguId) other; 

     return (this.getId()==castOther.getId() && this.getRevisiId()==castOther.getRevisiId()) 
&& (this.getRevisiId()==castOther.getRevisiId()); 
    } 

    public int hashCode() { 
     int result = 17; 

     result = 37 * result + this.getId(); 
     result = 37 * result + this.getRevisiId(); 
     return result; 
    } 


} 

当我试图挽救这个在数据库中,我得到了错误:

org.hibernate.NonUniqueObjectException: a different object with the same identifier value was already associated with the session: 

- 更新 - 但改变使用EmbeddedId的实现像这样

public class Pagu implements java.io.Serializable { 


    private PaguId id; 
    ... 

    @EmbeddedId 
@AttributeOverrides({ 
    @AttributeOverride(name="id", [email protected](name="id", nullable=false)), 
    @AttributeOverride(name="revisiId", [email protected](name="revisi_id", nullable=false)) }) 
public PaguId getId() { 
    return this.id; 
} 

public void setId(PaguId id) { 
    this.id = id; 
} 
.... 

它编译正确,但给我错误时,坚持模型。

org.hibernate.id.IdentifierGenerationException: ids for this class must be manually assigned before calling save(): id.model.Pagu 
+0

您不应该同时使用'@ Embeddable'和'@ IdClass'。可以选择在实体类的复合主键类和'@ EmbeddedId'上使用:'@ Embeddable',或者2.在复合主键类和'@ IdClass'上没有关于多个实体类级别的注释实体类本身的“@ Id”注释(用于字段)。 –

+0

这样做也会产生异常'具有相同标识符值的不同对象已经与会话相关联:[id.go.model.Pagu#id.go..model.PaguId @ 5ae9]'如果有的话,我得到的那种错误是@id – ahmy

+0

上的@GeneratedValue我找到了创建部分标识符生成的链接http://docs.jboss.org/hibernate/annotations/3.5/reference/en/html_single/#d0e1150 但还有一个警告那么您可能有兴趣创建自己的ID生成机制,这样的构造根本错误 – ahmy

回答

3

我不认为有可能在组合键中使用GeneratedValue,您必须选择组合键或单个GeneratedValue-id。

0

你必须从你的主要实体类中删除这两个钥匙ID和revisiId因为它已经存在于@Embeddable,尝试和分享您的答复。

+0

你可以找到一个工作示例[at](http://j2eereference.com/2011/01/implementing-composit-primary-key-with-jpa-and-hibernate/),很好的解释。 – mprabhat

+0

您可以将您的代码与我以前的评论中提到的代码进行比较(http://j2eereference.com/2011/01/implementing-composit-primary-key-with-jpa-and-hibernate/)。 – mprabhat

+0

我更新了代码以符合您的示例并添加@EmbeddableId。这是我使用netbeans从数据库逆向工程模型时的原始形式 – ahmy