嗨,我有这个大型 oracle hibernate web 应用程序,它似乎给出了这个错误
ORA-01795: maximum number of expressions in a list is 1000
我需要一个由某人测试的 java 代码作为休眠用户定义的组件,以便尽可能容易地添加到我的屏幕中的搜索 java 类中,有人可以拥有这样的测试组件吗?
嗨,我有这个大型 oracle hibernate web 应用程序,它似乎给出了这个错误
ORA-01795: maximum number of expressions in a list is 1000
我需要一个由某人测试的 java 代码作为休眠用户定义的组件,以便尽可能容易地添加到我的屏幕中的搜索 java 类中,有人可以拥有这样的测试组件吗?
我从链接中尝试了下面的代码,它似乎工作得很好,我将粘贴代码以防将来链接被破坏。
保持简单保持微笑:)
/**
* An utility method to build the Criterion Query IN clause if the number of parameter
* values passed has a size more than 1000. Oracle does not allow more than
* 1000 parameter values in a IN clause. Doing so a {@link SQLException} is
* thrown with error code, 'ORA-01795: maximum number of expressions in a list is 1000'.
* @param propertyName
* @param values
* @return
*/
import java.util.List;
import org.hibernate.criterion.Restrictions;
/**
*
* @author 2796
*/
public class SplitHibernateIn {
private static int PARAMETER_LIMIT = 999;
public static org.hibernate.criterion.Criterion buildInCriterion(String propertyName, List values) {
org.hibernate.criterion.Criterion criterion = null;
int listSize = values.size();
for (int i = 0; i < listSize; i += PARAMETER_LIMIT) {
List subList;
if (listSize > i + PARAMETER_LIMIT) {
subList = values.subList(i, (i + PARAMETER_LIMIT));
} else {
subList = values.subList(i, listSize);
}
if (criterion != null) {
criterion = Restrictions.or(criterion, Restrictions.in(propertyName, subList));
} else {
criterion = Restrictions.in(propertyName, subList);
}
}
return criterion;
}
}
相同的想法,但使用 javax Predicate。
private static int PARAMETER_LIMIT = 999;
private static Predicate createInStatement(CriteriaBuilder cb, Path fieldName, List values) {
int listSize = values.size();
Predicate predicate = null;
for (int i = 0; i < listSize; i += PARAMETER_LIMIT) {
List subList;
if (listSize > i + PARAMETER_LIMIT) {
subList = values.subList(i, (i + PARAMETER_LIMIT));
} else {
subList = values.subList(i, listSize);
}
if (predicate == null) {
predicate = fieldName.in(subList);
} else {
predicate = cb.or(predicate, fieldName.in(subList));
}
}
return predicate;
}
以及用法
public List<Bean> getBeanList(List<Long> pkList) {
CriteriaBuilder cb = entityManager.getCriteriaBuilder();
CriteriaQuery<Bean> query = cb.createQuery(Bean.class);
Root<Bean> root = query.from(Bean.class);
// "Bean_" is a @StaticMetamodel(Bean.class)
Predicate inStatement = createInStatement(cb, root.get(Bean_.pk), pkList);
query.select(root).where(inStatement);
return entityManager.createQuery(query).getResultList();
}
虽然目前的答案很好,但我认为这个答案更容易实现和理解:
private <T> Disjunction restrictionPropertyIn(String property, ArrayList<T> list) {
Disjunction criterion = Restrictions.disjunction();
for (List<T> idSubset : Lists.partition(list, 1000)) {
criterion.add(Restrictions.in(property, idSubset));
}
return criterion;
}
Restrictions.disjunction()
相当于Criteria
使用Restrictions.or()
.Lists
是来自 Guava 的实用程序类;partition(list, limit)
拆分list
为 size 的子列表limit
。返回Criterion
的可以在任何Criterion
预期 a 的地方按原样使用,例如:
List<Long> fiveThousandIds = Arrays.asList(1, 2, 3, ..., 999, 1000, 1001, ..., 5000);
Criteria crit = session.createCriteria(Employee.class);
crit.add(restrictionPropertyIn("employeeId", fiveThousandIds));
crit.list();
如果您需要支持具有不同IN
子句限制的不同数据库,您可以将硬编码1000
转换为参数。