Использование @ElementCollection в CriteriaQuery (или запрос содержимого @ElementCollection) - PullRequest
6 голосов
/ 08 октября 2010
public enum ReportStatus { 
    SUCCCEED, FAILED;
}

public class Work {
    @ElementCollection
    @Enumerated(EnumType.STRING)
    List<ReportStatus> reportStatuses;
}

Учитывая следующую структуру, я хотел бы выполнить запрос, чтобы найти всю работу, отфильтрованную по reportStatuses. Он отлично работает со следующим синтаксисом hql:

public List<Long> queryHQL() {
    final String query = "SELECT w.id FROM Work w JOIN w.reportStatuses s WHERE s in (:rs)";

    final List<ReportStatus> reportStatuses = new ArrayList<ReportStatus>();
    reportStatuses.add(ReportStatus.FAILED);

    return this.entityManager.createQuery(query).setParameter("rs", reportStatuses).getResultList();
}

Но я бы хотел использовать API критериев (jpa2) и не могу понять, как это сделать. Вот моя самая близкая попытка, я думаю:

public List<Long> query() {
    final List<ReportStatus> reportStatuses = new ArrayList<ReportStatus>();
    reportStatuses.add(ReportStatus.FAILED);

    final CriteriaBuilder builder = this.entityManager.getCriteriaBuilder();

    final CriteriaQuery<Long> criteriaQuery = builder.createQuery(Long.class);
    final Root<Work> workModel = criteriaQuery.from(Work.class);

    final ListJoin<Work, ReportStatus> status = workModel.joinList("reportStatuses");

    final Predicate predicate = status.in(reportStatuses);

    criteriaQuery.where(predicate);
    criteriaQuery.select(workModel.<Long> get("id"));

    return this.entityManager.createQuery(criteriaQuery).getResultList();
}

Я также пытался использовать API критериев гибернации, но как jpa2 я не смог найти правильный синтаксис.

Ответы [ 3 ]

5 голосов
/ 09 декабря 2010

Я бы использовал следующий синтаксис CriteriaQuery для того же.

EntityManager em = entityManagerFactory.createEntityManager();

final List<ReportStatus> reportStatuses = new ArrayList<ReportStatus>();
reportStatuses.add(ReportStatus.FAILED);

final CriteriaBuilder builder = em.getCriteriaBuilder();

final CriteriaQuery<Long> criteriaQuery = builder.createQuery(Long.class);
final Root<Work> _work = criteriaQuery.from(Work.class);

/*final ListJoin<Work, ReportStatus> status = _work.joinList("reportStatuses");
final Predicate predicate = status.in(reportStatuses);
criteriaQuery.where(predicate);*/

final Expression<List<ReportStatus>> _status = _work.get(Work_.reportStatuses);
_status.in(reportStatuses);

criteriaQuery.select(_work.get(Work_.id));

List<Long> list =  em.createQuery(criteriaQuery).getResultList();

UPDATE

Спасибо, но мы не используем сгенерированная метамодель. Так к сожалению Я не могу попробовать с твоим ответом. (

Если вы не используете Метамодель, просто замените _work.get(Work_.reportStatuses) на _work.get("reportStatuses"). Это сработает. :)

3 голосов
/ 21 августа 2011

Вы можете создать этот HQL.

String query = "SELECT w.id FROM Work w, IN(w.reportStatuses) s WHERE s = :rs";
return this.entityManager.createQuery(query).setParameter("rs", ReportStatus.FAILED).getResultList();
1 голос
/ 05 июля 2011

Я в замешательстве. Вызов in(..) возвращает предикат, но, похоже, на самом деле не применяет его (кажется, он не интегрирован в запрос & mdash; по крайней мере, для меня он вернул все члены корня независимо от того, пересекаются ли их коллекции с reportStatuses. В журнале отладки показан простой запрос select distinct work0_.id as id18_ from Work work0_).

Кроме того, зачем помещать reportStatuses в список, если вызывающая сторона интересуется только теми, которые соответствуют одному значению? Как бы вы сделали запрос с использованием ReportStatus.FAILED вместо создания списка для него?

...