MappedSuperclass-更改子类中的SequenceGenerator

问题描述 投票:26回答:4

我在Hibernate中使用JPA2,并尝试为我的实体引入一个通用的基类。到目前为止看起来像这样:

@MappedSuperclass
public abstract class BaseEntity {

    @Id
    private Long id;

    @Override
    public int hashCode() {
        // ...
    }

    @Override
    public boolean equals(Object obj) {
        // ...
    }

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

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

但是,对于每个表,都有一个序列$entityname_seq,我想将其用作序列生成器。我该如何在子类中进行设置?我想我需要重写@GeneratedValue并使用@SequenceGenerator创建一个新的SequenceGenerator。

java java-ee jpa sequence mappedsuperclass
4个回答
29
投票

是的,有可能。您可以使用@SequenceGenerator注释覆盖默认的生成器名称。

  • 基类
    @MappedSuperclass
    public abstract class PersistentEntity implements Serializable
    {
        private static final long serialVersionUID = 1L;

        @Id
        @GeneratedValue(strategy = GenerationType.SEQUENCE, generator = "default_gen")
        protected Long id = 0L;

        public Long getId()
        {
            return id;
        }

        public void setId(Long id)
        { 
            this.id = id;
        }
    }
  • 序列(SQL)

    create sequence role_seq;
  • 派生类

    @Entity
    @Table(name = "role")
    @SequenceGenerator(name = "default_gen", sequenceName = "role_seq", allocationSize = 1)
    public class Role extends PersistentEntity implements Serializable
    {
        private static final long serialVersionUID = 1L;

        @NotNull
        @Size(max = 32)
        private String name;

        public String getName()
        {
             return name;
        }

        public void setName(String name)
        {
             this.name = name;
        }   
    }
  • 此方法在Hibernate 4.1.x中很好用,但是在EclipseLink 2.x中却没有。

edit

  • 根据评论,它似乎可以在EclipseLink 2.6.1-RC1上使用。

12
投票

在JPA中,注释不能完成。注释本身不能被覆盖。实体从MappedSuperClass继承所有映射信息。只有两个注释可用于重新定义从映射的超类继承的映射:

  1. AttributeOverride覆盖列映射和
  2. [AssociationOverride覆盖连接列/表。

它们都不对GeneratedValue有帮助。


4
投票

使用EclipseLink,您可以使用CustomizerDescriptorCustomizer接口定义了一种自定义有关jpa描述符(aka持久实体)的所有信息的方法。

public class SequenceCustomizer implements DescriptorCustomizer {

    @Override
    public void customize(ClassDescriptor descriptor) throws Exception {
        descriptor.setSequenceNumberName(descriptor.getTableName());
    }
}

以及在您映射的超类中:

@MappedSuperclass
@Customizer(SequenceCustomizer.class)
public abstract class AbstractEntity implements Serializable {
    ...
}

0
投票

我正在写这篇文章,因为它对接受的答案的评论太不可读了:

我有一个BaseEntity,其他所有实体都继承自此:

BaseEntity.java:

@MappedSuperclass
public abstract class BaseEntity {

    @Id
    @GeneratedValue(strategy = GenerationType.SEQUENCE, generator = "SEQ_ID")
    private Long id;

然后我有两个实体UserOrder,它们都从BaseEntity继承,同时还具有@SequenceGenerator注释:

User.java:

@SequenceGenerator(name = "SEQ_ID", sequenceName = "SEQ_USER", allocationSize = 1)
public class User extends BaseEntity { ... }

Order.java:

@SequenceGenerator(name = "SEQ_ID", sequenceName = "SEQ_ORDER", allocationSize = 1)
public class Order extends BaseEntity { ... }

[它至少在H2上以2个序列SEQ_USERSEQ_ORDERS起作用:

select SEQ_USER.nextval from dual;
select SEQ_ORDERS.nextval from dual;
© www.soinside.com 2019 - 2024. All rights reserved.