我的Oracle 11g数据库中有一个四列表,用于实现扩展表反模式。我注意到一些查询耗时很长,并努力创建更好的索引;它在交互式会话中很好,但使用Spring的NamedJdbcTemplate
仍然很慢。
考虑以下例程:
private void getObjectIds(ObjectDomain domain, HashMap<String, List<String>> dimensionMap)
throws SQLException {
String sql = "SELECT m2.OBJECT_ID"
+ " FROM MetaInformations m1, MetaInformations m2\n"
+ " WHERE m1.OBJECT_ID = m2.OBJECT_ID\n"
+ " AND m1.OBJECT_DOMAIN = :domain AND m1.KEY = :key1 AND\n"
+ " m1.REF_OBJ_VALUE IN (:values1)\n"
+ " AND m2.OBJECT_DOMAIN = :domain AND m2.KEY = :key2 AND\n"
+ " m2.REF_OBJ_VALUE IN (:values2)";
String sqlWithBind = "SELECT m2.OBJECT_ID\n"
+ " FROM MetaInformations m1, MetaInformations m2\n"
+ " WHERE m1.OBJECT_ID = m2.OBJECT_ID\n"
+ " AND m1.OBJECT_DOMAIN = ? AND m1.KEY = ? AND\n"
+ " m1.REF_OBJ_VALUE IN (?, ?, ?, ?)\n"
+ " AND m2.OBJECT_DOMAIN = ? AND m2.KEY = ? AND\n"
+ " m2.REF_OBJ_VALUE IN (?)";
// Prebuilding statement, no bind variables left
Stopwatch stopWatch2 = Stopwatch.createStarted();
Iterator<Entry<String, List<String>>> entries = dimensionMap.entrySet().iterator();
Entry<String, List<String>> entry1 = entries.next();
Entry<String, List<String>> entry2 = entries.next();
String prebuilt = sql.replace(":domain", "'" + domain + "'")
.replace(":key1", "'" + entry1.getKey() + "'")
.replace(":values1",
entry1.getValue().stream().map(s -> "'" + s + "'").collect(Collectors.joining(", ")))
.replace(":key2", "'" + entry2.getKey() + "'")
.replace(":values2",
entry2.getValue().stream().map(s -> "'" + s + "'").collect(Collectors.joining(", ")));
Set<Long> rs2 = extractIdSet(getNamedParameterJdbcTemplate().queryForRowSet(prebuilt, Collections.emptyMap()));
log.warn("Prebuilt took: {} ms", stopWatch2.elapsed(TimeUnit.MILLISECONDS));
// Simple JDBCTemplate with 9 bind parameters
Stopwatch stopWatch5 = Stopwatch.createStarted();
Set<Long> rs1 = extractIdSet(getJdbcTemplate().queryForRowSet(sqlWithBind,
domain.toString(),
entry1.getKey(),
entry1.getValue().get(0),
entry1.getValue().get(1),
entry1.getValue().get(2),
entry1.getValue().get(3),
domain.toString(),
entry2.getKey(),
entry2.getValue().get(0)));
log.warn("JdbcTemplate took: {} ms", stopWatch5.elapsed(TimeUnit.MILLISECONDS));
// Most beautiful: NamedJDBCTemplate
Stopwatch stopWatch3 = Stopwatch.createStarted();
Map<String, Object> paramMap = createNamedParameterMap(domain, dimensionMap);
Set<Long> rs3 = extractIdSet(getNamedParameterJdbcTemplate().queryForRowSet(sql, paramMap));
log.warn("NamedParameterJdbcTemplate took: {} ms", stopWatch3.elapsed(TimeUnit.MILLISECONDS));
}
以下是结果。确切的时间因运行而异,但始终保持在同一数量级。
JdbcTemplate
,性能降低为爬网,约4秒。NamedJdbcTemplate
,与案例2一样慢;这至少就不足为奇了,因为幕后NamedJdbcTemplate
会将我的查询用命名参数替换为等同于案例2的东西。它没有得到连接,因为它们都是从同一个连接池中获取的。它似乎不仅仅是queryForRowSet()
函数,因为它实际上也是在最快的情况下使用的。同样,它看起来与Spring的异常翻译或参与正在进行的交易没有任何关系,因为这也应该影响案例1。
最后,问题是:与没有绑定参数的普通语句相比,为什么Spring的JdbcTemplate
绑定参数在这种情况下非常慢?
答案 0 :(得分:0)
事实证明它既不是JdbcTemplate
也不是NamedJdbcTemplate
。它也不是PreparedStatement
与Statement
,即使后者是最快的。这只是因为正常的语句没有带有绑定参数。如果我有没有绑定参数的查询,它与原始JDBC和NamedJdbcTemplate
的速度大致相同。
我们的Oracle 11 g 只是用9个绑定参数为这个查询选择了一个糟糕的执行计划,无论实际参数是什么,它都会坚持下去。我不知道为什么也没有真正的DBA。
使用相同数据对PostgreSQL 9.3数据库进行测试表明,无论是否使用绑定参数,它都同样快速;使用开箱即用的Ubuntu安装。