JPA 2条件获取路径导航

Alf*_*rio 26 java jpa fetch jpa-2.0

使用JPA 2 Criteria Join方法,我可以执行以下操作:

    //Join Example (default inner join)
    int age = 25;
    CriteriaBuilder cb = entityManager.getCriteriaBuilder();
    CriteriaQuery<Team> c = cb.createQuery(Team.class);
    Root<Team> t = c.from(Team.class);
    Join<Team, Player> p = t.join(Team_.players);
    c.select(t).where(cb.equal(p.get(Player_.age), age));
    TypedQuery<Team> q = entityManager.createQuery(c);
    List<Team> result = q.getResultList();
Run Code Online (Sandbox Code Playgroud)

我怎么能用fetch方法做同样的事情,我期望Fetch接口有get路径导航方法,但它没有:

    //Fetch Join Example

    int age = 25;
    CriteriaBuilder cb = entityManager.getCriteriaBuilder();
    CriteriaQuery<Team> cq = cb.createQuery(Team.class);
    Root<Team> t = cq.from(Team.class);
    Fetch<Team,Player> p = t.fetch(Team_.players);
    cq.where(cb.equal(p.get(Player_.age), age)); //This leads to compilation error there is no such method get in interface Fetch
    TypedQuery<Team> q = entityManager.createQuery(cq);
    List<Team> result = q.getResultList();
Run Code Online (Sandbox Code Playgroud)

根据Hiberante Documentation fetch返回一个错误的Join对象. http://docs.jboss.org/hibernate/stable/entitymanager/reference/en/html/querycriteria.html#querycriteria-from-fetch

Dat*_*eus 19

同意你对这种方法的看法,以及你希望它允许你所说的这一事实.另一种选择是

Join<Team, Player> p = t.join(Team_.players);
t.fetch(Team_.players);
c.select(t).where(cb.equal(p.get(Player_.age), age));
Run Code Online (Sandbox Code Playgroud)

即做一个join(),fetch()为它添加一个,然后使用连接.这是不合逻辑的,只会增加JPA标准的不雅性,但无论如何,应该是一个变通方法

  • 显然它可以工作,但它会导致多次加入:Hibernate:选择......来自Team team0_ inner join球员players__ on team0_.id = players1_.team_id内部加入球员1_1_ on players1_.id = players1_1_.id内部加入球员球员2_ on team0_ .id = players2_.team_id内部加入人物player2_1_ on players2_.id = players2_1_.id其中players1_1_.age = 25 (8认同)
  • 当你使用`mssql`执行`order by`和`distinct(true)`时,这会中断 (4认同)
  • 另请参阅http://blogs.sun.com/ldemichiel/entry/jpa_next_thinking_about_the#comment-1291653518000,其中提到了问题.基本上是一个没有考虑过的API (3认同)
  • 正如阿尔弗雷多所说,这确实双重连接。 (2认同)

Edu*_*cio 10

它适用于我使用Hibernate Provider.

//Join Example (default inner join)

    int age = 25;
    CriteriaBuilder cb = entityManager.getCriteriaBuilder();
    CriteriaQuery<Team> c = cb.createQuery(Team.class);
    Root<Team> t = c.from(Team.class);

    // Join<Team, Player> p = t.join(Team_.players); 
    Join<Team, Player> p = (Join<Team, Player>)t.fetch(Team_.players); 

    c.select(t).where(cb.equal(p.get(Player_.age), age));
    TypedQuery<Team> q = entityManager.createQuery(c);
    List<Team> result = q.getResultList();
Run Code Online (Sandbox Code Playgroud)

当然,它可能会破坏可移植性,但在我们的例子中,我们一直在使用其他hibernate的独有功能.

*这很奇怪,因为hibernate文档没有显示这个例子.

掌握它看看这个界面.

/*
 * Hibernate, Relational Persistence for Idiomatic Java
 *
 * Copyright (c) 2010, Red Hat Inc. or third-party contributors as
 * indicated by the @author tags or express copyright attribution
 * statements applied by the authors.  All third-party contributions are
 * distributed under license by Red Hat Inc.
 *
 * This copyrighted material is made available to anyone wishing to use, modify,
 * copy, or redistribute it subject to the terms and conditions of the GNU
 * Lesser General Public License, as published by the Free Software Foundation.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
 * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU Lesser General Public License
 * for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with this distribution; if not, write to:
 * Free Software Foundation, Inc.
 * 51 Franklin Street, Fifth Floor
 * Boston, MA  02110-1301  USA
 */
package org.hibernate.ejb.criteria;

import javax.persistence.criteria.Fetch;
import javax.persistence.criteria.Join;

/**
 * Consolidates the {@link Join} and {@link Fetch} hierarchies since that is how we implement them.
 * This allows us to treat them polymorphically.
*
* @author Steve Ebersole
*/
public interface JoinImplementor<Z,X> extends Join<Z,X>, Fetch<Z,X>, FromImplementor<Z,X> {
    /**
     * {@inheritDoc}
     * <p/>
     * Refined return type
     */
    public JoinImplementor<Z,X> correlateTo(CriteriaSubqueryImpl subquery);
}
Run Code Online (Sandbox Code Playgroud)


Dar*_*mer 6

从 JPA 2.1 开始,动态实体图可用于在条件查询中进行提取,同时使用 join() 而不是 fetch()。从问题中的例子来看:

//Join Example (default inner join)
int age = 25;
CriteriaBuilder cb = entityManager.getCriteriaBuilder();
CriteriaQuery<Team> c = cb.createQuery(Team.class);
Root<Team> t = c.from(Team.class);
Join<Team, Player> p = t.join(Team_.players);
c.select(t).where(cb.equal(p.get(Player_.age), age));
TypedQuery<Team> q = entityManager.createQuery(c);
List<Team> result = q.getResultList();
Run Code Online (Sandbox Code Playgroud)

如果这:

TypedQuery<Team> q = entityManager.createQuery(c);
Run Code Online (Sandbox Code Playgroud)

替换为:

EntityGraph<Team> fetchGraph = getEntityManager().createEntityGraph(Team.class);
fetchGraph.addSubgraph(Team_.players);
TypedQuery<Team> q = entityManager.createQuery(c).setHint("javax.persistence.loadgraph", fetchGraph);
Run Code Online (Sandbox Code Playgroud)

那么所有玩家都会趋之若鹜。


小智 5

您所要做的就是以下几点:

1- 取。2-然后,沿着路径走到你想要的地方。

在你的情况下:

int age = 25;
CriteriaBuilder cb = entityManager.getCriteriaBuilder();
CriteriaQuery<Team> cq = cb.createQuery(Team.class);
Root<Team> t = cq.from(Team.class);
Fetch<Team,Player> p = t.fetch(Team_.players);
cq.where(cb.equal(t.get("player").get("age"), age)); 
Run Code Online (Sandbox Code Playgroud)

  • `t.get("player")` 不存在,因为 Team 实体只有players 属性(在我们的示例中也是 Team_.players)。所以我会得到“java.lang.IllegalArgumentException:无法根据路径解析属性[玩家]” (2认同)