Формулирование критериев JPA в выражении - PullRequest
1 голос
/ 16 января 2012

Я прошу помощи в понимании того, как сформулировать условие in, используя пакет javax.persistence.criteria.

Я создаю критерий поиска CriteriaQuery для класса контактов. Контакт может принадлежать 0 ко многим типам контактов. Критерии поиска могут включать в себя значение фамилии, тип контакта или оба.

Когда я пытаюсь это сделать:

Expression<ContactType> param = criteriaBuilder.parameter(ContactType.class);           
Expression<List<ContactType>> contactTypes = fromContact.get("contactTypes");   
Predicate newPredicate = param.in(this.getContactType(), contactTypes);

Я получаю:

org.apache.openjpa.persistence.ArgumentException: Cannot execute query; declared parameters "ParameterExpression<ContactType>" were not given values.  You must supply a value for each of the following parameters, in the given order: [ParameterExpression<ContactType>]

Мне не удалось найти хороший пример того, как это сделать. Любая помощь и руководство очень ценится. Полный код ниже.

public CriteriaQuery<Contact> getSearchCriteriaQuery(EntityManager entityManager) {
    CriteriaBuilder criteriaBuilder = entityManager.getCriteriaBuilder();
    CriteriaQuery<Contact> criteriaQuery = criteriaBuilder.createQuery(Contact.class);
    Root<Contact> fromContact = criteriaQuery.from(Contact.class);
    Predicate whereClause = criteriaBuilder.equal(fromContact.get("domain"), this.getDomain());

    if (!StringUtils.isEmpty(this.getLastName())) {
        Predicate newPredicate = criteriaBuilder.equal(fromContact.get("lastName"), this.getLastName());
        whereClause = criteriaBuilder.and(whereClause, newPredicate);
    }

    if (this.getContactType() != null) {
        Expression<ContactType> param = criteriaBuilder.parameter(ContactType.class);
        Expression<List<ContactType>> contactTypes = fromContact.get("contactTypes");
        Predicate newPredicate = param.in(this.getContactType(), contactTypes);
        whereClause = criteriaBuilder.and(whereClause, newPredicate);
    }

    return criteriaQuery.where(whereClause);
}

@Entity
@Table(name = "contact")
public class Contact implements Serializable {

    private static final long serialVersionUID = -2139645102271977237L;
    private Long id;
    private String firstName;
    private String lastName;
    private Domain domain;
    private List<ContactType> contactTypes;

    public Contact() {
    }

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    @Column(unique = true, nullable = false)
    public Long getId() {
        return this.id;
    }

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

    @Column(name = "FIRST_NAME", length = 20)
    public String getFirstName() {
        return this.firstName;
    }

    public void setFirstName(String firstName) {
        this.firstName = firstName;
    }

    @Column(name = "LAST_NAME", length = 50)
    public String getLastName() {
        return this.lastName;
    }

    public void setLastName(String lastName) {
        this.lastName = lastName;
    }

    //bi-directional many-to-one association to Domain
    @ManyToOne(fetch = FetchType.LAZY)
    @JoinColumn(name = "DOMAIN")
    public Domain getDomain() {
        return this.domain;
    }

    public void setDomain(Domain domain) {
        this.domain = domain;
    }

    @ManyToMany
    @JoinTable(name = "CONTACT_CNTTYPE",
    joinColumns = {
        @JoinColumn(name = "CONTACT", referencedColumnName = "ID")},
    inverseJoinColumns = {
        @JoinColumn(name = "CONTACT_TYPE", referencedColumnName = "ID")})
    public List<ContactType> getContactTypes() {
        return this.contactTypes;
    }

    public void setContactTypes(List<ContactType> contactTypes) {
        this.contactTypes = contactTypes;
    }
}

1 Ответ

1 голос
/ 18 мая 2012

Вы должны установить значение параметра для запроса при выводе результатов:

TypedQuery<Entity> q = this.entityManager.createQuery(criteriaQuery);
q.setParameter(ContactType.class, yourContactTypeValueToFilter);
q.getResultList();

Что

criteriaBuilder.parameter(ContactType.class);

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

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