【问题标题】:PreparedStatement with Statement.RETURN_GENERATED_KEYS带有 Statement.RETURN_GENERATED_KEYS 的 PreparedStatement
【发布时间】:2011-05-12 13:46:11
【问题描述】:

某些 JDBC 驱动程序返回 Statement.RETURN_GENERATED_KEYS 的唯一方法是执行以下操作:

long key = -1L;
Statement statement = connection.createStatement();
statement.executeUpdate(YOUR_SQL_HERE, Statement.RETURN_GENERATED_KEYS);
ResultSet rs = statement.getGeneratedKeys();
if (rs != null && rs.next()) {
    key = rs.getLong(1);
}

有没有办法对PreparedStatement做同样的事情?


编辑

我询问是否可以对 PreparedStatement 执行相同操作的原因考虑以下情况:

private static final String SQL_CREATE = 
            "INSERT INTO
            USER(FIRST_NAME, MIDDLE_NAME, LAST_NAME, EMAIL_ADDRESS, DOB) 
            VALUES (?, ?, ?, ?, ?)";

USER 表中有一个PRIMARY KEY (USER_ID),它是一个BIGINT AUTOINCREMENT(因此您在SQL_CREATE 字符串中看不到它。

现在,我使用PreparedStatement.setXXXX(index, value) 填充?。我想返回ResultSet rs = PreparedStatement.getGeneratedKeys()。我怎样才能做到这一点?

【问题讨论】:

  • 很多人误解并使用 PreparedStatement#executeUpdate(arg) 。 Java 文档说This method with argument cannot be called on a PreparedStatement or CallableStatement. 这意味着我们必须使用不带参数的executeUpdate() 即使executeUpdate(arg) 方法可以在PreparedStatement 类中继承但我们不必使用它,否则我们会得到SQLException。

标签: java jdbc


【解决方案1】:

我现在没有编译器,我会通过提问来回答:

你试过吗?有用吗?

long key = -1L;
PreparedStatement statement = connection.prepareStatement();
statement.executeUpdate(YOUR_SQL_HERE, PreparedStatement.RETURN_GENERATED_KEYS);
ResultSet rs = statement.getGeneratedKeys();
if (rs != null && rs.next()) {
    key = rs.getLong(1);
}

免责声明:显然,我没有编译这个,但你明白了。

PreparedStatementStatement 的子接口,所以我看不出这不起作用的原因,除非某些 JDBC 驱动程序有问题。

【讨论】:

  • 这不是我要找的我知道PreparedStatementStatement 的子类....查看我更新的帖子。
【解决方案2】:

您可以使用 prepareStatement 方法获取额外的 int 参数

PreparedStatement ps = con.prepareStatement(sql, Statement.RETURN_GENERATED_KEYS)

对于某些 JDBC 驱动程序(例如 Oracle),您必须明确列出生成键的列名或索引:

PreparedStatement ps = con.prepareStatement(sql, new String[]{"USER_ID"})

【讨论】:

  • 我接受了您的回答,因为您展示了更多实现相同结果的方法。
【解决方案3】:

你的意思是这样的?

long key = -1L;

PreparedStatement preparedStatement = connection.prepareStatement(YOUR_SQL_HERE, PreparedStatement.RETURN_GENERATED_KEYS);
preparedStatement.setXXX(index, VALUE);
preparedStatement.executeUpdate();

ResultSet rs = preparedStatement.getGeneratedKeys();

if (rs.next()) {
    key = rs.getLong(1);
}

【讨论】:

  • 生成的keys结果集怎么可能为null?
【解决方案4】:
private void alarmEventInsert(DriveDetail driveDetail, String vehicleRegNo, int organizationId) {

    final String ALARM_EVENT_INS_SQL = "INSERT INTO alarm_event (event_code,param1,param2,org_id,created_time) VALUES (?,?,?,?,?)";
    CachedConnection conn = JDatabaseManager.getConnection();
    PreparedStatement ps = null;
    ResultSet generatedKeys = null;
    try {
        ps = conn.prepareStatement(ALARM_EVENT_INS_SQL, ps.RETURN_GENERATED_KEYS);
        ps.setInt(1, driveDetail.getEventCode());
        ps.setString(2, vehicleRegNo);
        ps.setString(3, null);
        ps.setInt(4, organizationId);
        ps.setString(5, driveDetail.getCreateTime());
        ps.execute();
        generatedKeys = ps.getGeneratedKeys();
        if (generatedKeys.next()) {
            driveDetail.setStopDuration(generatedKeys.getInt(1));
        }
    } catch (SQLException e) {
        e.printStackTrace();
        logger.error("Error inserting into alarm_event : {}", e
                .getMessage());
        logger.info(ps.toString());
    } finally {
        if (ps != null) {
            try {

                if (ps != null)
                    ps.close();
            } catch (SQLException e) {
                logger.error("Error closing prepared statements : {}", e
                        .getMessage());
            }
        }
    }
    JDatabaseManager.freeConnection(conn);
}

【讨论】:

  • 你不应该在 finally 块中释放你的连接,而不是在它之外(如果你遇到任何类型的运行时异常,你会窃取连接)?
  • @niraj - 我们可以写 Statement.RETURN_GENERATED_KEYS 而不是 ps.RETURN_GENERATED_KEYS 因为它是 java.sql.Statement 类中的静态变量。
【解决方案5】:
String query = "INSERT INTO ....";
PreparedStatement preparedStatement = connection.prepareStatement(query, PreparedStatement.RETURN_GENERATED_KEYS);

preparedStatement.setXXX(1, VALUE); 
preparedStatement.setXXX(2, VALUE); 
....
preparedStatement.executeUpdate();  

ResultSet rs = preparedStatement.getGeneratedKeys();  
int key = rs.next() ? rs.getInt(1) : 0;

if(key!=0){
    System.out.println("Generated key="+key);
}

【讨论】:

  • 如果生成了key,那么key,否则key=0,如果没有生成
猜你喜欢
  • 1970-01-01
  • 2012-12-02
  • 2011-12-05
  • 1970-01-01
  • 2013-05-02
  • 1970-01-01
  • 2021-05-17
  • 1970-01-01
相关资源
最近更新 更多