JPA 2 Критерии Выборка Навигация по пути - PullRequest
26 голосов
/ 22 декабря 2010

С помощью метода 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();

Как я могу сделать то же самое с методом выборки, я ожидал, что интерфейс 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();

Согласно Hiberante Documentation fetch возвращает объект Join, который является неправильным. http://docs.jboss.org/hibernate/stable/entitymanager/reference/en/html/querycriteria.html#querycriteria-from-fetch

Ответы [ 6 ]

19 голосов
/ 22 декабря 2010

Согласна с вами об этом методе и о том, что вы ожидаете, что он позволит то, что вы говорите. Другой вариант будет

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

Т.е. сделать join(), добавить для него fetch(), а затем использовать объединение. Это нелогично и только дополняет неэкспонентную природу критериев JPA, но в любом случае это должен быть обходной путь

10 голосов
/ 28 июня 2012

Это работает для меня, используя 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();

Конечно, это может нарушить переносимость, но в нашем случае мы использовали эксклюзивные функции других 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);
}
4 голосов
/ 16 октября 2015

Все, что вам нужно сделать, это следующее:

1- Do Fetch. 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)); 
3 голосов
/ 23 марта 2017

Начиная с 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();

Если это:

TypedQuery<Team> q = entityManager.createQuery(c);

заменяется следующим:

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

тогда все игроки будут с нетерпением ждать.

1 голос
/ 11 декабря 2015

Я использую JPA 2.1 с Hibernate 4.3.7, и нижеприведенное мне хорошо работает. Это даже не выглядит так некрасиво.

Join<Team,Player> p = (Join) t.fetch(Team_.players);
0 голосов
/ 07 апреля 2012

безобразно, но:

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

создаст сингл-соединение с fetch в sql, но это уродливый хак, который работает в спящем режиме JBoss6.1

...