首页 > 解决方案 > 使用 AttributeConverter 收集日志 ERROR HHH015007: Illegal argument on static metamodel field injection

问题描述

我正在使用以下转换器将一组字符串作为逗号分隔值(带有转义)保存在单个VARCHAR数据库列中:

public class StringSetConverter implements AttributeConverter<Set<String>, String> {

   @Override
   public String convertToDatabaseColumn(Set<String> stringSet) {
      if (stringSet == null) {
         return null;
      }
      // remove null-strings and empty strings, escape commas
      return stringSet.stream().filter(Objects::nonNull).filter(s -> s.length() > 0)
                    .map(s -> s.replace(",", "\\,"))
                    .sorted().collect(Collectors.joining(","));
   }

   @Override
   public Set<String> convertToEntityAttribute(String dbString) {
      if (dbString == null) {
        return new HashSet<>();
      }
      Set<String> result = new HashSet<>();
      String[] items = dbString.split("(?<=[^\\\\]),");
      for (String item : items) {
         if (item.length() > 0) {
            result.add(item.replace("\\,", ","));
         }
      }
      return result;
   }
}

JPA 实体上的用法:

    @Column(name = "GRANTED_PRIVILEGES", nullable = false, length = 4000)
    @Convert(converter = StringSetConverter.class)
    private Set<String> grantedPrivileges;

    @Column(name = "DENIED_PRIVILEGES", nullable = false, length = 4000)
    @Convert(converter = StringSetConverter.class)
    private Set<String> deniedPrivileges;

转换器工作得非常好- 字符串集得到正确持久化,可以毫无问题地读回。

唯一需要注意的是Hibernate在启动使用这些实体的 Spring Boot 应用程序时记录的错误:

    ERROR o.h.m.i.MetadataContext : HHH015007: Illegal argument on static metamodel field injection : 
        anonymized.AuthorityDO_#grantedPrivileges; 
        expected type :  org.hibernate.metamodel.internal.SingularAttributeImpl; 
        encountered type : javax.persistence.metamodel.SetAttribute 
    ERROR  o.h.m.i.MetadataContext : HHH015007: Illegal argument on static metamodel field injection : 
        anonymized.AuthorityDO_#deniedPrivileges; 
        expected type :  org.hibernate.metamodel.internal.SingularAttributeImpl; 
        encountered type : javax.persistence.metamodel.SetAttribute

Hibernate 对我的 JPA 有什么问题AttributeConverter?除了记录错误之外,Hibernate 最终可以很好地与转换器配合使用——那么问题出在哪里,我该如何解决呢?

标签: hibernatespring-bootjpaspring-data-jpa

解决方案


您的映射是错误的,因为您尝试将集合用于单数属性。根据 JPA 规范(参见6.2.1.1 规范元模型部分):

z对于由 class 声明的每个持久集合值属性X,其中元素类型zZ,元模型类必须包含如下声明:

  • 如果集合类型zjava.util.Set,那么

    public static volatile SetAttribute<X, Z> z;

但是,hibernate 期望您将@ElementCollection,@OneToMany@ManyToMany关联用于集合属性。

我建议您为您的自定义类型使用一些包装类:

public class StringSet {
  private Set<String> stringSet;
  // ...
}

public class StringSetConverter implements AttributeConverter<StringSet, String> {

    @Override
    public String convertToDatabaseColumn(StringSet stringSet) {
        // ...
    }

    @Override
    public StringSet convertToEntityAttribute(String dbString) {
       // ...
    }
}

@Column(name = "GRANTED_PRIVILEGES", nullable = false, length = 4000)
@Convert(converter = StringSetConverter.class)
private StringSet grantedPrivileges;

推荐阅读