Как создать и обработать составной первичный ключ в JPA

108

Я хочу иметь версии из той же записи данных. Другими словами, я хочу продублировать запись с другим номером версии.

id - Version будет первичным ключом.

Как должна выглядеть сущность? Как его дублировать с другой версией?

id Version ColumnA

1   0      Some data
1   1      Some Other data
2   0      Data 2. Entry
2   1      Data
Кайзер
источник
При использовании @IdClassаннотации я обнаружил еще один совет: @Columnаннотация должна входить в поля класса Entity ( YourEntityв примере кода RohitJan).
KenSV

Ответы:

231

Вы можете создать Embedded class, который содержит ваши два ключа, а затем иметь ссылку на этот класс, как EmbeddedIdв вашем Entity.

Вы должны бы @EmbeddedIdи @Embeddableаннотации.

@Entity
public class YourEntity {
    @EmbeddedId
    private MyKey myKey;

    @Column(name = "ColumnA")
    private String columnA;

    /** Your getters and setters **/
}
@Embeddable
public class MyKey implements Serializable {

    @Column(name = "Id", nullable = false)
    private int id;

    @Column(name = "Version", nullable = false)
    private int version;

    /** getters and setters **/
}

Еще один способ для достижения этой задачи является использование @IdClassаннотаций и место как ваш idв этом IdClass. Теперь вы можете использовать обычную @Idаннотацию для обоих атрибутов.

@Entity
@IdClass(MyKey.class)
public class YourEntity {
   @Id
   private int id;
   @Id
   private int version;

}

public class MyKey implements Serializable {
   private int id;
   private int version;
}
Рохит Джайн
источник
4
Можно ли использовать @Generatedvalueдля идентификаторов EmbeddedId
Kayser
1
@Kayser. Насколько я знаю. Нет. Вы должны явно установить для них значение в экземпляре KeyClass, а затем установить этот экземпляр класса ключа в своем Entity.
Рохит Джайн,
@Kayser. @GeneratedValueможет использоваться только для генерации ключевых значений для первичного ключа, он не может генерировать комбинацию для составных ключей.
Рохит Джайн,
1
@RohitJain только одно: на самом деле вы не можете сделать встроенный класс общедоступным (он должен быть в отдельном файле, чтобы быть общедоступным)
Лукас,
1
@FastEngy К нему по-прежнему можно получить доступ через Wayback Machine: web.archive.org/web/20170123035517/http://uaihebert.com/… . Похоже, что эта статья заменена на web.archive.org/web/20170202203555/http://uaihebert.com/… и web.archive.org/web/20161014051056/http://uaihebert.com/…, которые также исчезли …
радлан 05
9

Класс MyKey должен быть реализован, Serializableесли вы используете@IdClass

Свапнил17
источник
5

Ключевой класс:

@Embeddable
@Access (AccessType.FIELD)
public class EntryKey implements Serializable {

    public EntryKey() {
    }

    public EntryKey(final Long id, final Long version) {
        this.id = id;
        this.version = version;
    }

    public Long getId() {
        return this.id;
    }

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

    public Long getVersion() {
        return this.version;
    }

    public void setVersion(Long version) {
        this.version = version;
    }

    public boolean equals(Object other) {
        if (this == other)
            return true;
        if (!(other instanceof EntryKey))
            return false;
        EntryKey castOther = (EntryKey) other;
        return id.equals(castOther.id) && version.equals(castOther.version);
    }

    public int hashCode() {
        final int prime = 31;
        int hash = 17;
        hash = hash * prime + this.id.hashCode();
        hash = hash * prime + this.version.hashCode();
        return hash;
    }

    @Column (name = "ID")
    private Long id;
    @Column (name = "VERSION")
    private Long operatorId;
}

Класс сущности:

@Entity
@Table (name = "YOUR_TABLE_NAME")
public class Entry implements Serializable {

    @EmbeddedId
    public EntryKey getKey() {
        return this.key;
    }

    public void setKey(EntryKey id) {
        this.id = id;
    }

    ...

    private EntryKey key;
    ...
}

Как я могу продублировать его с другой версией?

Вы можете отсоединить объект, полученный от поставщика, изменить ключ Entry и затем сохранить его как новый объект.

callfarc0de
источник
Можно указать идентификатор в Entrykey AUTOGENERATED. или что-то в этом роде @GeneratedValue(strategy = GenerationType.IDENTITY)
Kayser
1
Мне также интересно, как рассчитать хеш для двух длинных первичных ключей. Что же касается hashи primeв методе hashCodeв классе EntryKey, вы можете сказать мне , где эта идея исходит от?
Брюс Сан
1

Класс MyKey (@Embeddable) не должен иметь отношений вроде @ManyToOne

Ранука
источник
Почему нет? Вы видели этот пример ?
Buhake Sindi