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()
);

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

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,我们知道基本类型不能为空,你会有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

JdbcTemplate#queryForInt如果列值为SQL NULL或0,则返回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;
}

最新更新