如何对 ElementCollection 和 CollectionTable 使用 CriteriaQuery
How to use CriteriaQuery for ElementCollection and CollectionTable
我有一个非常简单的实体产品,它有代码、名称和标签。标签存储在另一个 table (product_tag) 中,带有 product_id 和标签列。
我需要使用 CriteriaQuery 搜索带有特定标签的产品。举个例子,我想找到具有 'fruit' 和 'red' 标签的产品。
使用 spring 4.1.x、spring-data-jpa 1.8 和休眠 4.2.x.
我的实体就是;
@Entity
@Table(name = "product", uniqueConstraints ={
@UniqueConstraint(columnNames = "code")
}
)
@NamedQueries({
@NamedQuery(name = "Product.findAll", query = "select p from Product p")
})
public class Product extends EntityWithId {
@Column(name = "code", length = 128)
private String code;
@Column(name = "name", length = 512)
protected String name;
@ElementCollection(fetch = FetchType.EAGER)
@CollectionTable(name="product_tag", joinColumns=@JoinColumn(name="product_id"))
@Column(name="tag")
private Set<String> productTags = new HashSet<>();
}
这是我如何启动搜索的代码;
private void search() {
CriteriaBuilder builder = entityManager.getCriteriaBuilder();
CriteriaQuery<Product> criteriaQuery = builder.createQuery(Product.class);
Root<Product> product = criteriaQuery.from(Product.class);
Predicate where = builder.conjunction();
if (!StringUtils.isEmpty(nameSearch.getValue())) {
where = builder.and(where, builder.like(product.<String>get("name"), nameSearch.getValue() + "%"));
}
if (!StringUtils.isEmpty(codeSearch.getValue())) {
where = builder.and(where, builder.like(product.<String>get("code"), codeSearch.getValue() + "%"));
}
if (!StringUtils.isEmpty(tagsSearch.getValue())) {
//Util.parseCommaSeparated returns Set<String>
where = builder.and(where, product.get("productTags").in(Util.parseCommaSeparated(tagsSearch.getValue())));
}
criteriaQuery.where(where);
List<Product> resultList = entityManager.createQuery(criteriaQuery).getResultList();
}
然而,当我 运行 搜索标签时 'fruit' 我得到一个异常
java.lang.IllegalArgumentException: Parameter value [fruit] did not match expected type [java.util.Set (n/a)]
我真的很想知道对 ElementCollection 和 CollectionTable 使用 CriteriaQuery。
productTags
映射到单独的 table,因此您需要在查询中加入 table。
...
if (!StringUtils.isEmpty(tagsSearch.getValue())) {
//Util.parseCommaSeparated returns Set<String>
where = builder.and(where, product.join("productTags").in(Util.parseCommaSeparated(tagsSearch.getValue())));
}
...
注意 product.join("productTags")
而不是 product.get("productTags")
尝试使用 isMember() 而不是 in()
检查 the example 5 和 7
我有一个非常简单的实体产品,它有代码、名称和标签。标签存储在另一个 table (product_tag) 中,带有 product_id 和标签列。
我需要使用 CriteriaQuery 搜索带有特定标签的产品。举个例子,我想找到具有 'fruit' 和 'red' 标签的产品。
使用 spring 4.1.x、spring-data-jpa 1.8 和休眠 4.2.x.
我的实体就是;
@Entity
@Table(name = "product", uniqueConstraints ={
@UniqueConstraint(columnNames = "code")
}
)
@NamedQueries({
@NamedQuery(name = "Product.findAll", query = "select p from Product p")
})
public class Product extends EntityWithId {
@Column(name = "code", length = 128)
private String code;
@Column(name = "name", length = 512)
protected String name;
@ElementCollection(fetch = FetchType.EAGER)
@CollectionTable(name="product_tag", joinColumns=@JoinColumn(name="product_id"))
@Column(name="tag")
private Set<String> productTags = new HashSet<>();
}
这是我如何启动搜索的代码;
private void search() {
CriteriaBuilder builder = entityManager.getCriteriaBuilder();
CriteriaQuery<Product> criteriaQuery = builder.createQuery(Product.class);
Root<Product> product = criteriaQuery.from(Product.class);
Predicate where = builder.conjunction();
if (!StringUtils.isEmpty(nameSearch.getValue())) {
where = builder.and(where, builder.like(product.<String>get("name"), nameSearch.getValue() + "%"));
}
if (!StringUtils.isEmpty(codeSearch.getValue())) {
where = builder.and(where, builder.like(product.<String>get("code"), codeSearch.getValue() + "%"));
}
if (!StringUtils.isEmpty(tagsSearch.getValue())) {
//Util.parseCommaSeparated returns Set<String>
where = builder.and(where, product.get("productTags").in(Util.parseCommaSeparated(tagsSearch.getValue())));
}
criteriaQuery.where(where);
List<Product> resultList = entityManager.createQuery(criteriaQuery).getResultList();
}
然而,当我 运行 搜索标签时 'fruit' 我得到一个异常
java.lang.IllegalArgumentException: Parameter value [fruit] did not match expected type [java.util.Set (n/a)]
我真的很想知道对 ElementCollection 和 CollectionTable 使用 CriteriaQuery。
productTags
映射到单独的 table,因此您需要在查询中加入 table。
...
if (!StringUtils.isEmpty(tagsSearch.getValue())) {
//Util.parseCommaSeparated returns Set<String>
where = builder.and(where, product.join("productTags").in(Util.parseCommaSeparated(tagsSearch.getValue())));
}
...
注意 product.join("productTags")
而不是 product.get("productTags")
尝试使用 isMember() 而不是 in()
检查 the example 5 和 7