jpa критерии для многих ко многим отношениям - PullRequest
16 голосов
/ 15 ноября 2011

У меня есть 2 класса POJO на Java, Answer и Collaborator, в отношении многие ко многим.

class Answer {
    @ManyToMany(cascade = CascadeType.ALL)
    @JoinTable(name = "ANSWERS_COLLABORATORS", joinColumns = { @JoinColumn(name = "aid") }, inverseJoinColumns = { @JoinColumn(name = "cid") })
    private Set<Collaborator> collaborators = new HashSet<Collaborator>(0);
} 

Класс Answer имеет набор Collaborator, но Collaboratorне хранит набор Answer.Что мне нужно сделать из Hibernate CriteriaQuery, так это найти соавторов для ответа, заданного id.

Я уже сделал это с Hibernate Criteria (org.hibernate.Criteria), используя преобразователь результатов, но я 'Я застрял, когда дело доходит до использования CriteriaQuery, потому что у меня нет списка ответов, которые можно дать объединению.

Ответы [ 2 ]

21 голосов
/ 18 ноября 2011

Готово, наконец-то ...

Вот код:

public List<Collaborator> getCollaborators(Long answerId) {
        CriteriaBuilder criteriaBuilder = entityManager.getCriteriaBuilder();
        CriteriaQuery<Collaborator> criteriaQuery = criteriaBuilder
                .createQuery(Collaborator.class);
        Root<Answer> answerRoot = criteriaQuery.from(Answer.class);
        criteriaQuery.where(criteriaBuilder.equal(answerRoot.get(Answer_.id),
                answerId));
        SetJoin<Answer, Collaborator> answers = answerRoot
                .join(Answer_.collaborators);
        CriteriaQuery<Collaborator> cq = criteriaQuery.select(answers);
        TypedQuery<Collaborator> query = entityManager.createQuery(cq);
        return query.getResultList();

    }
8 голосов
/ 15 ноября 2011

Использование HQL:

Вы можете использовать это:

Criteria criteria = session.createCriteria(Answer.class);
criteria.setResultTransformer(Criteria.DISTINCT_ROOT_ENTITY);
criteria.createAlias("collaborators", "collaborators");
criteria.add(Restrictions.eq("collaborators.id",desiredCollaboratorId);

, чтобы получить все ответы, связанные с определенным коллаборатором.

И это:

Criteria criteria = session.createCriteria(Answer.class);
criteria.setResultTransformer(Criteria.DISTINCT_ROOT_ENTITY);
criteria.setFetchMode("collaborators", FetchMode.JOIN)
criteria.add(Restrictions.idEq(desiredAnswerId));
dsrTrackingCriteria.setProjection(Projections.property("collaborators"));

Чтобы получить всех коллабораторов, связанных с определенным ответом.

Используя API критериев JPA2, вы можете сделать что-то вроде:

CriteriaBuilder cb = em.getCriteriaBuilder(); //creted from EntityManager instance

CriteriaQuery<Long> cq = cb.createQuery(Collaborator.class);
Root<Answer> rootAnswer = cq.from(Answer.class);
Join<Collaborator,Answer> joinAnswerCollaborator = rootAnswer.join("collaborators"); //(or rootAnswer.join(Answer_.collaborators); if you've created the metamodel with JPA2
...