在Java中,我试图从ResultSet测试空值,其中列被转换为原始int类型。

int iVal;
ResultSet rs = magicallyAppearingStmt.executeQuery(query);
if (rs.next()) {
  if (rs.getObject("ID_PARENT") != null && !rs.wasNull()) {
    iVal = rs.getInt("ID_PARENT");
  }
}

从上面的代码片段中,是否有更好的方法来做到这一点,并且我假设第二个wasNull()测试是多余的?

教育我们,谢谢


当前回答

在Kotlin中,我只会解决它一次,并永远解决这个问题:

fun <K : Any> ResultSet.getNullable(columnLabel: String, type: KClass<K>): K? = 
    this.getObject(columnLabel, type.java)

之后你可以这样做:

rs.getNullable("ID_PARENT", Int::class)

我想如果你愿意,你也可以这样做

fun <K> ResultSet.getNullable(columnLabel: String, type: Class<K>): K? = 
    this.getObject(columnLabel, type)

所以你可以这样做:

rs.getNullable("ID_PARENT", Int::class.java)

如果你遇到的开发人员连最简单的事情都不能达成一致,最好还是同时提供两种方法。

fun <K : Any> ResultSet.getNullable(columnLabel: String, type: KClass<K>): K? =
        this.getNullable(columnLabel, type.java)

fun <K> ResultSet.getNullable(columnLabel: String, type: Class<K>): K? =
        this.getObject(columnLabel, type)

编辑:如果库仍然是挑剔的,你终于可以这样做:

rs.getNullable("ID_PARENT", String::class)?.let {FOO.valueOf(it) }

其他回答

以防有人在Kotlin编程时来到这里(就像我一样),BalusC建议的答案可以很好地工作。请注意,Short和Float在ResultSet中分别被实例化为Integer和Double,并且在调用getObject()后,我们应该将它们强制转换为正确的类型。在我的案例中,最终代码是:

when {
    propKClass.isSubclassOf(Int::class) -> rs.getObject(colName) as Int? 
    propKClass.isSubclassOf(Short::class) -> (rs.getObject(colName) as Int?)?.toShort()
    propKClass.isSubclassOf(Long::class) -> rs.getObject(colName) as Long?
    propKClass.isSubclassOf(Boolean::class) -> rs.getObject(colName) as Boolean?
    propKClass.isSubclassOf(Double::class) -> rs.getObject(colName) as Double?
    propKClass.isSubclassOf(Float::class) -> (rs.getObject(colName) as Double?)?.toFloat()
    else -> rs.getString(colName)
}

ResultSet的默认值。当字段值为NULL时,getInt将返回0,这也是iVal声明的默认值。在这种情况下,您的测试是完全多余的。

如果你真的想做一些不同的事情,如果字段值为NULL,我建议:

int iVal = 0;
ResultSet rs = magicallyAppearingStmt.executeQuery(query);
if (rs.next()) {
    iVal = rs.getInt("ID_PARENT");
    if (rs.wasNull()) {
        // handle NULL field value
    }
}

(编辑如下@martin评论;写的OP代码不会编译,因为iVal没有初始化)

在Kotlin中,我只会解决它一次,并永远解决这个问题:

fun <K : Any> ResultSet.getNullable(columnLabel: String, type: KClass<K>): K? = 
    this.getObject(columnLabel, type.java)

之后你可以这样做:

rs.getNullable("ID_PARENT", Int::class)

我想如果你愿意,你也可以这样做

fun <K> ResultSet.getNullable(columnLabel: String, type: Class<K>): K? = 
    this.getObject(columnLabel, type)

所以你可以这样做:

rs.getNullable("ID_PARENT", Int::class.java)

如果你遇到的开发人员连最简单的事情都不能达成一致,最好还是同时提供两种方法。

fun <K : Any> ResultSet.getNullable(columnLabel: String, type: KClass<K>): K? =
        this.getNullable(columnLabel, type.java)

fun <K> ResultSet.getNullable(columnLabel: String, type: Class<K>): K? =
        this.getObject(columnLabel, type)

编辑:如果库仍然是挑剔的,你终于可以这样做:

rs.getNullable("ID_PARENT", String::class)?.let {FOO.valueOf(it) }

如果你想要一个替代调用ResultSet.wasNull()的方法,你可以使用getObject()并强制转换为正确的类型。

Long val = (Long)rs.getObject(pos++);

你也可以用setObject()在语句中设置空值。

pstmt.setObject(pos++, null);

我想你可以简单地使用

iVal = rs.getInt("ID_PARENT");
if (rs.wasNull()) {
  // do somthing interesting to handle this situation
}

即使它是NULL。