hibernate未绑定类型,使用泛型时没有显式的目标实体

5f0d552i  于 2021-07-09  发布在  Java
关注(0)|答案(2)|浏览(305)

我有以下两门课:

@Entity 
@Table(name = "criteria") 
@Inheritance(strategy = InheritanceType.SINGLE_TABLE)
@DiscriminatorColumn(name = AbstractCriteria.TYPE, discriminatorType = DiscriminatorType.STRING)
public abstract class AbstractCriteria<T> implements Serializable {

   public static final String TYPE = "type";

   ...
   @Column(name = "clazz") private Class<T> clazz;
   ...

   protected AbstractCriteria(Class<T> clazz) {
      this.clazz = Preconditions.checkNotNull(clazz);
   }
   ...

   public abstract Predicate<T> evaluate(T value);
}

@Entity 
@DiscriminatorValue(CriteriaType.Values.DATETIME_IS_BETWEEN) 
public class DateTimeIsBetweenCriteria extends AbstractCriteria<DateTime> {
   ...
   public DateTimeIsBetweenCriteria() {
      super(DateTime.class);
   }
   ...

   @Override 
   public Predicate<DateTime> evaluate(DateTime value) {
      ...
   }
}

当我试图保存datetimeisbetweencriteria类的示例时,出现以下异常:

Exception in thread "main" org.hibernate.AnnotationException: 
Property ...AbstractCriteria.clazz has an unbound type and no explicit target entity.
Resolve this Generic usage issue or set an explicit target attribute (eg @OneToMany(target=) or use an explicit @Type

我需要 clazz 对于中的方法 AbstractCriteria ,和 T 可以是字符串、日期时间、布尔值等。
我在考虑
clazz @Transient ,但如果这样做,我仍然需要另一个列来标识要保存的条件,以便在从数据库加载条件时,我知道它是哪种条件。我觉得这有点麻烦。
有什么想法我可以解决这个例外而不改变我的设计?
非常感谢你抽出时间。

zed5wv10

zed5wv101#

我已将实现更改为不使用参数化类型:

@Entity 
@Table(name = "criteria") 
@Inheritance(strategy = InheritanceType.SINGLE_TABLE)
@DiscriminatorColumn(name = AbstractCriteria.TYPE, discriminatorType = DiscriminatorType.STRING)
public abstract class AbstractCriteria implements Serializable {

  public static final String TYPE = "type";

  ...
  @Column(name = "clazz") private Class clazz;
  ...

  protected AbstractCriteria(Class clazz) {
     this.clazz = Preconditions.checkNotNull(clazz);
  }
  ...

  public abstract Predicate evaluate(Object value);
}

@Entity 
@DiscriminatorValue(CriteriaType.Values.DATETIME_IS_BETWEEN) 
public class DateTimeIsBetweenCriteria extends AbstractCriteria {
  ...
  public DateTimeIsBetweenCriteria() {
     super(DateTime.class);
  }
  ...

  @Override 
  public Predicate evaluate(Object value) {
     if (value instanceof DateTime) {
        // do evaluation and return here.
     } else {
        throw new UserDefinedException("...");
     }
  }
}
oipij1gg

oipij1gg2#

就泛型而言,您的思路是正确的,但是您正在尝试持久化“class”类型的类。数据库没有可以直接Map到的数据类型(因此是您的错误),我能想到的最接近的方法是使用字符串作为类名;

@Column (name="className")
private String classname;

并设置

.... setClassname (DateTime.getCanonicalName());

检索类;

Class.forName(classname);

但是如果我对您正在做的事情有正确的想法,那么您根本不需要持久化这个类:所有datetimeisbetweencriteria示例与operation on-datetime-它是类的模板参数,而不是示例。你应该坚持你的约会间隔,而不是班级的行为。

相关问题