Statement.RETURN_GENERATED_KEYSを使用したPreparedStatement


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 2015年

回答:


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 2010

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 2010

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 2017

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ブロックで解放するべきではありません(何らかのランタイム例外が発生した場合は、接続を解放します)。
ジュール

@ niraj-ps.RETURN_GENERATED_KEYSの代わりに、java.sql.Statementクラスの静的変数であるStatement.RETURN_GENERATED_KEYSを記述できます。
AmitG 2015年
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.