我们经常会有这种业务需求,根据一个条件集合去查询一张表的数据,比如:
select * from all_element t where t.task_id in (List <taskids>);
在java语言中,我们需要用到JDBC来和数据库打交道,那么在JDBC中该如何处理这种需求呢?我们可以有如下几种处理方式
方案一:写一个函数把参数集合转换成一个or 条件 或 in 条件的字符串,最后拼成一个sql
select * from all_element t where t.task_id in (123 ,456, 789);
或者是:
select * from all_element t where t.task_id=123 or t.task_id= 456 t.task_id= 789;
但是这样效率如何呢?我们知道Oracle对传过来的SQL是需要事先编译的,不过是Oracle有个缓存功能可以缓存编译好的SQL,但前提是传过来的SQL必须完全一致,很明显,如果按照以上方式的话,一旦taskid值变化,那么Oracle的缓存便无法利用。
方案二:使用预编译的PrepareStatement
为了解决Oracle缓存无法利用问题,Jdbc提供了预编译的PrepareStatement,对于变化的参数可以用占位符 <?> 来代替,因此我们可以用占位符来减少Oracle编译次数。
private static final String QUERY = "select * from all_element where taskId = ?";
ps = con .prepareStatement(QUERY);
for(String taskId : taskIds){
ps.setInt(1, taskId);
rs = ps .executeQuery();
}
这样做虽然可以很好的利用Oracle的缓存,但缺点也很明显,就是每一个Id都需要查询数据库一次,这样效率是极低的。
方案三:动态地创建PrepareStatement
虽然变化的参数可以用占位符 <?> 来代替,然而遗憾的是Jdbc只提供了单一占位符功能即占位符不能是一个可迭代的集合。因此,对于我们传过来的集合参数,我们可以动态地创建一个PrepareStatement:
拼一个和集合大小相等数量占位符的SQL,再写一个循环来赋值每一个占位符,这样就可以解决taskId的值变化而导致Oracle重新编译SQL问题。
private static void createQuery(List<String> taskIds) {
String query = "select * from all_element t where t.task_id in (";
StringBuilder queryBuilder = new StringBuilder(query);
for (int i = 0; i < taskIds.size(); i++) {
queryBuilder.append(" ?");
if (i != taskIds.size() - 1)
queryBuilder.append(",");
}
queryBuilder.append(")");
ps = con.prepareStatement(queryBuilder.toString());
for (int i = 1; i <= taskIds.size(); i++) {
ps.setInt(i, taskIds.get(i - 1));
}
rs = ps.executeQuery();
}
标签:JDBC,批量,占位,task,SQL,Oracle,操作,where,id From: https://blog.51cto.com/u_16421711/9101173