JdbcTemplate queryForInt / Long在Spring 3.2.2中已弃用。应该用什么代替?

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了JdbcTemplate queryForInt / Long在Spring 3.2.2中已弃用。应该用什么代替?相关的知识,希望对你有一定的参考价值。

JdbcTemplate中的queryforInt / queryforLong方法在Spring 3.2中已弃用。我无法找出使用这些方法替换现有代码的最佳做法的原因或内容。

一种典型的方法:

int rowCount = jscoreJdbcTemplate.queryForInt(
    "SELECT count(*) FROM _player WHERE nameKey = ? AND teamClub = ?",
    playerNameKey.toUpperCase(),
    teamNameKey.toUpperCase()
);

好的,上面的方法需要重写如下:

Object[] params = new Object[] { 
   playerNameKey.toUpperCase(), 
   teamNameKey.toUpperCase()
};
int rowCount = jscoreJdbcTemplate.queryForObject(
    "SELECT count(*) FROM _player WHERE nameKey = ? AND teamClub = ?",
    params, Integer.class);

显然,这种弃用使JdbcTemplate类更简单(或者它?)。 QueryForInt总是一种方便的方法(我猜)并且已经存在了很长时间。为什么删除它。结果代码变得更加复杂。

答案

我认为有人意识到queryForInt / Long方法具有令人困惑的语义,也就是说,从JdbcTemplate源代码中可以看到它的当前实现:

@Deprecated
public int queryForInt(String sql, Object... args) throws DataAccessException {
    Number number = queryForObject(sql, args, Integer.class);
    return (number != null ? number.intValue() : 0);
}

这可能会导致您认为如果结果集为空它将返回0,但它会抛出异常:

org.springframework.dao.EmptyResultDataAccessException: Incorrect result size: expected 1, actual 0

所以以下实现基本上等同于当前的实现:

@Deprecated
public int queryForInt(String sql, Object... args) throws DataAccessException {
    return queryForObject(sql, args, Integer.class);
}

然后,现在必须用丑陋的代码替换未弃用的代码:

    queryForObject(sql, new Object { arg1, arg2, ...}, Integer.class);

或者这个(更好):

    queryForObject(sql, Integer.class, arg1, arg2, ...);
另一答案

我同意原来的海报,即弃用方便方法queryForLong(sql)是不方便的。

我使用Spring 3.1开发了一个应用程序,并且刚刚更新到最新的Spring版本(3.2.3),并注意到它已被弃用。

幸运的是,这对我来说是一个改变:

return jdbcTemplate.queryForLong(sql);  // deprecated in Spring 3.2.x

被改为

return jdbcTemplate.queryForObject(sql, Long.class);

并且几个单元测试似乎表明,上述变化有效。

另一答案

贬低赞成queryForObject(String, Class)

另一答案

替换此类代码:

long num = jdbcTemplate.queryForLong(sql);

使用此代码:

long num = jdbcTemplate.queryForObject(sql, Long.class);

是非常危险的,因为如果列具有空值,则queryForObject返回null并且我们知道原始类型不能为null并且您将具有NullPointerException。编译器没有警告你这件事。您将在运行时了解此错误。如果您有返回基本类型的方法,则会出现相同的错误:

public long getValue(String sql) {
    return = jdbcTemplate.queryForObject(sql, Long.class);
}

Spring 3.2.2中JdbcTemplate中不推荐使用的方法queryForLong具有以下主体:

@Deprecated
public long queryForLong(String sql) throws DataAccessException {
    Number number = queryForObject(sql, Long.class);
    return (number != null ? number.longValue() : 0);
}

你看,在它们返回原始值之前,检查它是否为null,如果它为null则返回0.顺便说一下 - 应该是0L。

另一答案

如果列值为SQL NULL或0,则JdbcTemplate#queryForInt返回0.无法区分一个案例与另一个案例。我认为这是该方法被弃用的主要原因。顺便说一下,ResultSet#getInt的行为与此类似。虽然,我们可以通过ResultSet#wasNull来区分这两种情况。

另一答案
public int getCircleCount() {
    Object param = "1";
    String sql = "select count(*) from circle where id = ? ";
    jdbcTemplate.setDataSource(getDataSource());
    int result = getJdbcTemplate().queryForObject(sql, new Object[] { param }, Integer.class);
    return result;
}

以上是关于JdbcTemplate queryForInt / Long在Spring 3.2.2中已弃用。应该用什么代替?的主要内容,如果未能解决你的问题,请参考以下文章

Spring的jdbcTemplate使用

jdbcTemplate.queryForXXX等方法的总结

Hibernate JdbcTemplate的queryForInt的些许问题

Spring Jdbc

JdbcTemplate查询数据 三种callback之间的区别

spring jdbcTemplate查询使用