为啥我不能使用与 JPA 2.1 中的实体类型匹配的 TYPE-WHERE 子句查询实体?

Posted

技术标签:

【中文标题】为啥我不能使用与 JPA 2.1 中的实体类型匹配的 TYPE-WHERE 子句查询实体?【英文标题】:Why can't I query an entity with a TYPE-WHERE clause matching the entity type in JPA 2.1?为什么我不能使用与 JPA 2.1 中的实体类型匹配的 TYPE-WHERE 子句查询实体? 【发布时间】:2016-03-28 18:18:03 【问题描述】:

以实体类A的形式查询

SELECT a from A a WHERE TYPE(a) = A

失败

could not resolve property: class of: de.richtercloud.type.operator.nonsense.A [SELECT a from de.richtercloud.type.operator.nonsense.A a WHERE TYPE(a) = A]

我不明白,因为将A 限制为A 应该排除所有不是As 的超类实例以及子类实例。

例子:

package de.richtercloud.type.operator.nonsense;

import java.io.File;
import java.sql.SQLException;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.Persistence;
import javax.persistence.Query;

/**
 * Illustrates the problem/misunderstanding that storing an @link A and
 * querying it with a @code WHERE TYPE([identifier]) = A fails with
 * @code org.hibernate.QueryException: could not resolve property:
 * class of: de.richtercloud.type.operator.nonsense.A.
 * @author richter
 */
public class NewMain 
    private final static File DATABASE_DIR = new File("/tmp/type-operator-nonsense");
    private final static String DERBY_CONNECTION_URL = String.format("jdbc:derby:%s", DATABASE_DIR.getAbsolutePath());

    /**
     * @param args the command line arguments
     */
    public static void main(String[] args) throws SQLException 
        //setup database
        EntityManagerFactory entityManagerFactory = null;
        try 
            Map<Object, Object> entityManagerFactoryMap = new HashMap<>();
            entityManagerFactoryMap.put("javax.persistence.jdbc.url",
                    String.format("%s;create=%s", DERBY_CONNECTION_URL, !DATABASE_DIR.exists()));
            entityManagerFactory = Persistence.createEntityManagerFactory("type-operator-nonsense",
                    entityManagerFactoryMap);

            //show issue
            EntityManager entityManager = entityManagerFactory.createEntityManager();
            A a = new A(1L, "b");
            entityManager.getTransaction().begin();
            entityManager.persist(a);
            entityManager.flush();
            Query query = entityManager.createQuery("SELECT a from A a WHERE TYPE(a) = A");
            List<?> queryResult = query.getResultList();
            entityManager.getTransaction().commit();
            System.out.println(queryResult.size());
        finally 
            if(entityManagerFactory != null) 
                entityManagerFactory.close();
            
        
    

persistence.xml:

<?xml version="1.0" encoding="UTF-8"?>
<persistence version="2.1" xmlns="http://xmlns.jcp.org/xml/ns/persistence" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/persistence http://xmlns.jcp.org/xml/ns/persistence/persistence_2_1.xsd">
  <persistence-unit name="type-operator-nonsense" transaction-type="RESOURCE_LOCAL">
    <provider>org.hibernate.ejb.HibernatePersistence</provider>
    <class>de.richtercloud.type.operator.nonsense.A</class>
    <properties>
      <property name="javax.persistence.jdbc.url" value="jdbc:derby:/tmp/type-operator-nonsense"/>
      <property name="javax.persistence.jdbc.user" value=""/>
      <property name="javax.persistence.jdbc.driver" value="org.apache.derby.jdbc.EmbeddedDriver"/>
      <property name="javax.persistence.jdbc.password" value=""/>
      <property name="hibernate.cache.provider_class" value="org.hibernate.cache.NoCacheProvider"/>
      <property name="hibernate.hbm2ddl.auto" value="update"/>
    </properties>
  </persistence-unit>
</persistence>

org.hibernate:hibernate-entitymanager:5.1.0.Finalorg.apache.derby:derby:10.11.1.1

A 在此示例中是一个 POJO,它不涉及继承,这与我期望的限制无关(尽管如果没有继承,应用它就没有意义):

package de.richtercloud.type.operator.nonsense;

import java.io.Serializable;
import javax.persistence.Entity;
import javax.persistence.Id;

@Entity
public class A implements Serializable 
    private static final long serialVersionUID = 1L;
    private String b;
    @Id
    private Long id;

    public A() 
    

    public A(Long id, String b) 
        this.id = id;
        this.b = b;
    

    public void setB(String b) 
        this.b = b;
    

    public String getB() 
        return b;
    

    public Long getId() 
        return id;
    

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

【问题讨论】:

我只是探索了这个问题,发现很少有人愿意与您分享。 您能否尝试使用 .class 而不是 Type() "SELECT a from A a WHERE a.class= A" 或者为实体类 A 创建一个存储库并使用您的实际查询访问 Type -SELECT a from A a WHERE TYPE(a) = A SELECT a from A a WHERE a.class = A 失败并出现完全相同的错误,这表明这可能是休眠错误或无意义的反馈。 “存储库”是什么意思 - jar 依赖项 - 说 maven 术语?如果是,那为什么会有什么不同? 我的意思是 JPA repsitory 和休眠的 .class 表示法。请参考链接可能对您有用***.com/questions/4884249/… 当我在 GitHub 上查看您的示例时 - @Inheritance 不应该在 A 上吗?因为 B 没有子类,因此注释没有意义。看看下面我的项目示例。 【参考方案1】:

此行为是由于休眠错误https://hibernate.atlassian.net/browse/HHH-10653 造成的。上面的示例适用于 OpenJPA。

【讨论】:

【参考方案2】:

TYPE() 只能用于继承映射。

A 是继承映射中的一个类,A 是带有 @Inheritance 的路由类还是 A 扩展了一个声明了 @Inheritance 的类?

示例。

@Inheritance
@Entity
public class Project

@Entity
public class DesignProject extends Project

@Entity
public class QualityProject extends Project

@Entity
public class SoftwareProject extends Project

现在你可以使用 TYPE()

SELECT p
FROM Project p
WHERE TYPE(p) = DesignProject OR TYPE(p) = QualityProject

【讨论】:

能否加个参考或详细说明。为什么JPA 2.1 specification 中没有提到这一点?你的意思是“根类”,对吧?我在不涉及实体继承的情况下遇到此问题,请参阅更新的问题了解详细信息。 我可以通过将继承层次结构中最顶层的类设为MappedSuperclass 来触发问题。我不明白你所说的@Inheritance 声明的意思,因为有一个默认值InheritanceType.SINGLE_TABLE,即你的陈述只有在你提到一个具体的策略时才有意义(并解释它会导致什么麻烦)。我用所有三种策略验证了这个问题。 我的意思是你必须将它与继承树中的实体一起使用。 @MappedSupperclass 在这种情况下不是继承。 SELECT a from A a WHERE TYPE(a) = A 在您的示例中只是没有意义,因为 A 没有继承。所以A永远是A 在下面查看我的答案。

以上是关于为啥我不能使用与 JPA 2.1 中的实体类型匹配的 TYPE-WHERE 子句查询实体?的主要内容,如果未能解决你的问题,请参考以下文章

JPA 2.1 中的 @ConstructorResult 映射不能与 Hibernate 4.3.5.Final 一起正常工作

JPA 2.1 Eager Fetch 属性

如何使用Spring Data JPA在实体中映射Oracle Object类型?

为啥编译器告诉我没有运算符与我的“if”和“else if”语句中的操作数类型匹配?

未使用的 JPA 实体是不是被垃圾回收?为啥?

休眠查询异常:在 JPA 查询期间无法解析实体属性