带语句的PreparedStatement。RETURN_GENERATED_KEYS


83

一些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字符串。

现在,我填充?using PreparedStatement.setXXXX(index, value)。我想回来ResultSet rs = PreparedStatement.getGeneratedKeys()。我该如何实现?


2
许多人误解并使用PreparedStatement#executeUpdate(arg)。Java doc表示This method with argument cannot be called on a PreparedStatement or CallableStatement.这意味着即使executeUpdate(arg)方法可以在PreparedStatement类中继承,我们也必须使用不带参数的executeUpdate(),但是我们不必使用它,否则会得到SQLException。
AmitG'1

Answers:


141

您可以使用prepareStatement带有其他int参数的方法

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

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

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

我已经接受了您的回答,因为您展示了更多获得相同结果的方法。
Buhake Sindi

67

你的意思是这样的吗?

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

生成的键结果集如何为null?
AlikElzin-kilaka 2016年

10

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

你有试过吗 它行得通吗?

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....的子类。请参阅我的最新文章。
Buhake Sindi

2
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);
}

如果生成了密钥,则生成密钥,否则生成密钥= 0
Dharmendrasinh Chudasama

0
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);
}

1
您是否不应该在finally块中而不是在其外部释放连接(如果您遇到任何类型的运行时异常,您就会喜欢连接)?
2013年

@niraj-代替ps.RETURN_GENERATED_KEYS,我们可以编写Statement.RETURN_GENERATED_KEYS,因为它是java.sql.Statement类中的静态变量。
AmitG 2015年
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.