Как получить сгенерированные ключи из пакетной вставки JDBC в Oracle?

Я вставляю много записей, используя вставки JDBC. Есть ли способ получить сгенерированный ключ для каждой записи? Могу ли я использовать ps.getGeneratedKeys() с пакетными вставками?

Я использую oracle.jdbc.OracleDriver

final String insert = "Insert into Student(RollNumber, Name, Age) values(StudentSEQ.nextval, ? , ?)";
final int BATCH_SIZE = 998;
int count = 0;
Connection con = null;
PreparedStatement ps =  null;
try {
    con = getConnection();
    ps = con.prepareStatement(insert);
    for (Student s : students) {
        ps.setString(1, s.getName());
        ps.setInt(2, s.getAge());
        ps.addBatch();
        count++;
        if (count % BATCH_SIZE == 0) {
        // Insert records in batches
            ps.executeBatch();
        }
    }
    // Insert remaining records
    ps.executeBatch();
} finally {
    if(ps != null)
        ps.close();
    release(con);
}

Я думаю использовать ps.executeUpdate() вместе с ps.getGeneratedKeys() внутри цикла, чтобы получить желаемый результат. Любые другие решения?

Ответ 1

Похоже, что Oracle 12c не поддерживает объединение автоматически сгенерированных ключей с пакетным обновлением в соответствии со следующей страницей:

http://docs.oracle.com/cd/E16655_01/java.121/e17657/jdbcvers.htm

См. подраздел "Ограничения" в разделе "Извлечение автоматически сгенерированных ключей"

Ответ 2

спецификация JDBC 4.1, раздел 13.6 Извлечение значений автоматического генерирования говорит:

Определяется реализацией того, вернется ли getGeneratedKeysгенерируемые значения после вызова метода executeBatch.

Поэтому вам нужно будет проверить, действительно ли ваш драйвер поддерживает его для пакетных обновлений. Как указано в ответе Филиппа О., поиск сгенерированных ключей не поддерживается пакетными обновлениями, как описано в Поддержка стандартов Oracle 12 JDBC:

Вы не можете комбинировать автоматически сгенерированные ключи с пакетным обновлением.

В любом случае, если он поддерживается вашим драйвером, а подготовка вашего оператора должна быть изменена на следующий код, чтобы дать указание драйверу получить сгенерированные ключи:

ps = con.prepareStatement(insert, Statement.RETURN_GENERATED_KEYS);

Примечание: вам может понадобиться использовать один из методов подготовки сгенерированных ключей (prepareStatement(sql, columnIndexes) или prepareStatement(sql, columnNames)), поскольку Oracle вернет ROW_ID с помощью метода в моем примере.