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표 A 거기 PRIMARY KEY (USER_ID)a는이다 BIGINT AUTOINCREMENT(따라서 왜 당신은 그것을 볼 수 없습니다 SQL_CREATE문자열.

이제 ?사용하여 PreparedStatement.setXXXX(index, value). 나는 돌아가고 싶다 ResultSet rs = PreparedStatement.getGeneratedKeys(). 어떻게하면 되나요?


2
많은 사람들이 PreparedStatement # executeUpdate (arg)를 오해하고 사용합니다. Java doc에 따르면 PreparedStatement 클래스에서 메소드를 상속받을 수 This method with argument cannot be called on a PreparedStatement or CallableStatement.있지만 인수없이 executeUpdate ()를 사용해야 executeUpdate(arg)하지만 사용할 필요가 없습니다. 그렇지 않으면 SQLException이 발생합니다.
AmitG

답변:


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 17:02 :

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 드라이버가 버그가 아닌 경우 이것이 작동하지 않는 이유를 알 수 없습니다.


그것은 내가 찾고있는 것이 아닙니다 . ... PreparedStatement의 하위 클래스 라는 것을 알고 Statement있습니다. 업데이트 된 게시물을 참조하십시오.
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 블록에서 연결을 해제해야하지 않습니까 (어떤 종류의 런타임 예외가 발생하면 연결이 끊어 질 것입니다)?
Jules

@niraj-ps.RETURN_GENERATED_KEYS 대신 java.sql.Statement 클래스의 정적 변수이기 때문에 Statement.RETURN_GENERATED_KEYS를 작성할 수 있습니다.
AmitG
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.