PreparedStatement avec Statement.RETURN_GENERATED_KEYS

La seule façon pour certains pilotes JDBC de renvoyer Statement.RETURN_GENERATED_KEYS consiste à effectuer les opérations suivantes:

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

Est-il possible de faire la même chose avec PreparedStatement ?


modifier

La raison pour laquelle j’ai demandé si je pouvais faire la même chose avec PreparedStatement tient compte du scénario suivant:

 private static final Ssortingng SQL_CREATE = "INSERT INTO USER(FIRST_NAME, MIDDLE_NAME, LAST_NAME, EMAIL_ADDRESS, DOB) VALUES (?, ?, ?, ?, ?)"; 

Dans la table USER , il y a une PRIMARY KEY (USER_ID) qui est un BIGINT AUTOINCREMENT (d’où pourquoi vous ne le voyez pas dans la chaîne SQL_CREATE .

Maintenant, je peupler le ? en utilisant PreparedStatement.setXXXX(index, value) . Je veux retourner ResultSet rs = PreparedStatement.getGeneratedKeys() . Comment puis-je atteindre cet objective?

Vous pouvez soit utiliser la méthode prepareStatement prenant un paramètre int supplémentaire

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

Pour certains pilotes JDBC (par exemple, Oracle), vous devez lister explicitement les noms de colonne ou les index des clés générées:

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

Tu veux dire quelque chose comme ca?

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

N’ayant pas de compilateur par moi en ce moment, je répondrai en posant une question:

Avez-vous essayé cela? Est-ce que ça marche?

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

Disclaimer: Évidemment, je n’ai pas compilé cela, mais vous avez l’idée.

PreparedStatement est une sous-interface de Statement , donc je ne vois aucune raison pour laquelle cela ne fonctionnerait pas, à moins que certains pilotes JDBC ne soient bogués.

 Ssortingng 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); } 
 private void alarmEventInsert(DriveDetail driveDetail, Ssortingng vehicleRegNo, int organizationId) { final Ssortingng 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.setSsortingng(2, vehicleRegNo); ps.setSsortingng(3, null); ps.setInt(4, organizationId); ps.setSsortingng(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.toSsortingng()); } finally { if (ps != null) { try { if (ps != null) ps.close(); } catch (SQLException e) { logger.error("Error closing prepared statements : {}", e .getMessage()); } } } JDatabaseManager.freeConnection(conn); }