Product.java
@ManyToMany
@JoinTable(name = "product_has_warehouse", joinColumns = @JoinColumn(name = ID), inverseJoinColumns = @JoinColumn(
name = "warehouse_id"), foreignKey = @ForeignKey(name = FK + "has_warehouse"),
inverseForeignKey = @ForeignKey(name = FK + "warehouse"))
private List<Warehouse> warehouses;
因此,我需要进行条件查询以按产品ID获取仓库列表。
我试过了:
final CriteriaQuery<Warehouse> query = getCriteriaBuilder().createQuery(Warehouse.class);
final Root<Product> root = query.from(Product.class);
final Join<Product, Warehouse> warehouseJoin = root.join("warehouses");
query.where(getCriteriaBuilder().in(warehouseJoin));
final TypedQuery<Warehouse> typedQuery = getEm().createQuery(query);
return typedQuery.getResultList();
但我明白了:
java.lang.IllegalArgumentException:验证时出错 标准引起:java.lang.IllegalStateException:没有显式 选择和隐含的选择无法确定
答案 0 :(得分:3)
这应该有效:
@Test
public void getWarehousesByProduct() {
Product drinks = new Product("drinks");
Warehouse wh1 = new Warehouse("house1");
Warehouse wh2 = new Warehouse("house2");
Warehouse wh3 = new Warehouse("house3");
Warehouse wh4 = new Warehouse("house4");
Warehouse wh5 = new Warehouse("house5");
Warehouse wh6 = new Warehouse("house6");
drinks.getWarehouses().add(wh1);
drinks.getWarehouses().add(wh2);
drinks.getWarehouses().add(wh3);
drinks.getWarehouses().add(wh4);
saveAll(Arrays.asList(new Warehouse[]{wh1,wh2,wh3,wh4,wh5,wh6}));
em.persist(drinks);
CriteriaBuilder cb = em.getCriteriaBuilder();
CriteriaQuery<Warehouse> cq = cb.createQuery(Warehouse.class);
Root<Product> product = cq.from(Product.class);
Join<Product, Warehouse> warehouses = product.join("warehouses");
cq.select(warehouses).where(cb.equal(product.get("id"), drinks.getId()));
TypedQuery<Warehouse> tq = em.createQuery(cq);
List<Warehouse> result = tq.getResultList();
Assert.assertNotNull(result);
Assert.assertEquals(drinks.getWarehouses().size(), result.size());
}
请注意,我只是"one directional"
方式。以bidirectional
方式,您需要将产品添加到其可行的仓库(以满足双向条件)。
顺便说一句,你的错误遗漏了cq.select(warehouses)
这就是为什么你有一个非法的州,因为你的查询不知道该选择什么。