107

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 一直是一种方便的方法(我猜)并且已经存在了很长时间。为什么被删除了。结果代码变得更加复杂。

4

6 回答 6

112

我认为有人意识到 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, ...);
于 2013-04-05T16:40:01.743 回答
38

我同意原始发帖人的观点,即弃用便捷方法 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);

一些单元测试似乎表明,上述更改有效。

于 2013-07-25T15:09:51.273 回答
15

已弃用,取而代之的是queryForObject(String, Class).

于 2013-03-27T14:33:59.223 回答
13

替换这样的代码:

long num = jdbcTemplate.queryForLong(sql);

使用此代码:

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

非常危险,因为如果列有 null 值 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);
}

您会看到在它们返回原始值之前检查这是否为空,如果为空,则返回 0。顺便说一下 - 应该是 0L。

于 2013-08-06T09:22:43.013 回答
2

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

于 2016-03-17T16:23:38.083 回答
-1
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;
}
于 2018-03-05T06:36:07.167 回答