ПідготовленийЗвіт із заявою.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.

Тепер я заповнюю ?використання PreparedStatement.setXXXX(index, value). Я хочу повернутися ResultSet rs = PreparedStatement.getGeneratedKeys(). Як я можу цього досягти?


2
Багато людей неправильно розуміють і використовують PreparedStatement # executeUpdate (arg). Документ Java говорить, що This method with argument cannot be called on a PreparedStatement or CallableStatement.це означає, що ми повинні використовувати executeUpdate () без аргументу, хоча executeUpdate(arg)метод може бути успадкований у класі PreparedStatement, але нам не потрібно його використовувати, інакше ми отримаємо 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"})

Я прийняв вашу відповідь, оскільки ви показали більше способів досягти того самого результату.
Бухаке Сінді

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

Як може набір результатів згенерованих ключів бути нульовим?
AlikElzin-kilaka

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

Застереження: Очевидно, я не складав цього, але ви розумієте.

PreparedStatement є підінтерфейсом Statement , тому я не бачу причини, чому це не спрацювало б, якщо деякі драйвери JDBC не виправлені помилки.


це не те, що я шукаю. Я знаю, що PreparedStatementце підклас Statement.... див. мій оновлений пост.
Бухаке Сінді

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, якщо не сформований
Дхармендрасін Чудасама

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
Чи не слід звільняти підключення в блоці нарешті, а не за його межами (ви отримаєте підключення, якщо отримаєте будь-яке виключення під час виконання)?
Жуль

@niraj - замість ps.RETURN_GENERATED_KEYS ми можемо написати Statement.RETURN_GENERATED_KEYS, оскільки це статична змінна в класі java.sql.Statement.
AmitG
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.