Hibernate, наследование одной таблицы и использование поля из суперкласса в качестве столбца дискриминатора - PullRequest
9 голосов
/ 14 июля 2010

У меня есть следующие виды классов для иерархии спящих объектов. Я пытаюсь иметь два конкретных подкласса Sub1Class и Sub2Class. Они разделены столбцом дискриминатора (field), который определен в MappedSuperClass. Существует абстрактный класс сущностей EntitySuperClass, на который ссылаются другие сущности. Другие сущности не должны заботиться, ссылаются ли они на Sub1Class или Sub2Class.

Это действительно возможно? В настоящее время я получаю эту ошибку (потому что определение столбца наследуется дважды в Sub1Class и в EntitySuperClass):

Repeated column in mapping for entity: my.package.Sub1Class column: field (should be mapped with insert="false" update="false")

Если я добавлю @MappedSuperClass к EntitySuperClass, я получу ошибку подтверждения от hiberante: ему не нравится, если класс является одновременно Entity и сопоставленным суперклассом. Если я удаляю @Entity из EntitySuperClass, класс больше не является сущностью и на него нельзя ссылаться из других сущностей:

MappedSuperClass является частью внешнего пакета, поэтому по возможности его не следует изменять.

Мои занятия:

@MappedSuperclass
public class MappedSuperClass {
    private static final String ID_SEQ = "dummy_id_seq";
    @Id
    @GeneratedValue(strategy = GenerationType.SEQUENCE, generator = ID_SEQ)
    @GenericGenerator(name=ID_SEQ, strategy="sequence")

    @Column(name = "id", unique = true, nullable = false, insertable = true, updatable = false)
    private Integer id;

    @Column(name="field", nullable=false, length=8)
    private String field;

    public Integer getId() {
        return id;
    }
    public void setId(Integer id) {
        this.id = id;
    }
    public String getField() {
        return field;
    }
    public void setField(String field) {
        this.field = field;
    }
}


@Entity
@Table(name = "ACTOR")
@Inheritance(strategy=InheritanceType.SINGLE_TABLE)
@DiscriminatorColumn(name="field", discriminatorType=DiscriminatorType.STRING)
abstract public class EntitySuperClass extends MappedSuperClass {


    @Column(name="description", nullable=false, length=8)
    private String description;

    public String getDescription() {
        return description;
    }

    public void setDescription(String description) {
        this.description = description;
    }
}

@Entity
@DiscriminatorValue("sub1")
public class Sub1Class extends EntitySuperClass {

}


@Entity
@DiscriminatorValue("sub2")
public class Sub2Class extends EntitySuperClass {

}


@Entity
public class ReferencingEntity {
    @Id
    @GeneratedValue(strategy = GenerationType.SEQUENCE)
    private Integer id;

    @Column
    private Integer value;

    @ManyToOne
    private EntitySuperClass entitySuperClass;


    public Integer getId() {
        return id;
    }

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

    public Integer getValue() {
        return value;
    }

    public void setValue(Integer value) {
        this.value = value;
    }

    public EntitySuperClass getEntitySuperClass() {
        return entitySuperClass;
    }

    public void setEntitySuperClass(EntitySuperClass entitySuperClass) {
        this.entitySuperClass = entitySuperClass;
    }

}

Ответы [ 4 ]

15 голосов
/ 14 июля 2010

В моем проекте это делается так:

@Entity
@Inheritance(strategy = InheritanceType.SINGLE_TABLE)
@DiscriminatorColumn(name = "field", discriminatorType = DiscriminatorType.STRING)
@DiscriminatorValue("dummy")
public class EntitySuperClass {
    // here definitions go 
    // but don't define discriminator column here
}

@Entity
@DiscriminatorValue(value="sub1")
public class Sub1Class extends EntitySuperClass {
    // here definitions go
}

И это работает.Я думаю, что ваша проблема в том, что вы без необходимости определяете поле дискриминатора в своем определении суперкласса.Удалите его, и он будет работать.

13 голосов
/ 14 июля 2010

Чтобы использовать столбец дискриминатора в качестве обычного свойства, вы должны сделать это свойство доступным только для чтения с insertable = false, updatable = false. Поскольку вы не можете изменить MappedSuperClass, вам нужно использовать @AttributeOverride:

@Entity 
@Table(name = "ACTOR") 
@Inheritance(strategy=InheritanceType.SINGLE_TABLE) 
@DiscriminatorColumn(name="field", discriminatorType=DiscriminatorType.STRING) 

@AttributeOverride(name = "field", 
    column = @Column(name="field", nullable=false, length=8, 
        insertable = false, updatable = false))

abstract public class EntitySuperClass extends MappedSuperClass { 
    ...
}
2 голосов
/ 15 июля 2010

Вы можете сопоставить столбец базы данных только один раз как поле для чтения-записи (поле, которое имеет insertable=true и / или updatable=true) и любое количество раз как поле только для чтения (insertable=false и * 1005).* updatable=false).Использование столбца в качестве @DiscriminatorColumn считается отображением для чтения и записи, поэтому у вас не может быть дополнительных сопоставлений для чтения и записи.

Hibernate установит значение, указанное в @DiscriminatorColumn, за кулисами на основе конкретного экземпляра класса,Если бы вы могли изменить это поле, это позволило бы изменить поле @DiscriminatorColumn, чтобы ваш подкласс и значение в поле могли не совпадать.

0 голосов
/ 23 ноября 2013

Один фундаментальный: вам фактически не нужно извлекать столбец дискриминатора из БД.У вас уже должна быть та информация в коде, которую вы используете в своих тегах @DiscriminatorValue.Если вам нужно прочитать это из БД, тщательно пересмотрите способ назначения дискриминаторов.

Если вам это нужно в конечном объекте сущности, хорошей практикой может быть реализация значения Enum from дискриминатора и возврат его в@ Переходное поле:

@Entity
@Table(name="tablename")
@DiscriminatorValue(Discriminators.SubOne.getDisc())
public class SubClassOneEntity extends SuperClassEntity {

    ...

    @Transient
    private Discriminators discriminator;

    // Setter and Getter
    ...
}

public enum Discriminators {
     SubOne ("Sub1"),
     SubOne ("Sub2");

     private String disc;
     private Discriminators(String disc) { this.disc = disc; }
     public String getDisc() { return this.disc; }
}
...