带有 Statement.RETURN_GENERATED_KEYS 的 PreparedStatement
Posted
技术标签:
【中文标题】带有 Statement.RETURN_GENERATED_KEYS 的 PreparedStatement【英文标题】:PreparedStatement with Statement.RETURN_GENERATED_KEYS 【发布时间】:2011-05-12 13:46:11 【问题描述】:某些 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()
。我怎样才能做到这一点?
【问题讨论】:
很多人误解和使用 PreparedStatement#executeUpdate(arg) 。 Java 文档说This method with argument cannot be called on a PreparedStatement or CallableStatement.
这意味着我们必须使用不带参数的executeUpdate() 即使executeUpdate(arg)
方法可以在PreparedStatement 类中继承但我们不必使用它,否则我们会得到SQLException。
【参考方案1】:
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);
【讨论】:
如果生成了key,那么key,否则key=0,如果没有生成【参考方案2】:你的意思是这样的?
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);
【讨论】:
生成的keys结果集怎么可能为null?【参考方案3】: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);
【讨论】:
你不应该在 finally 块中释放你的连接,而不是在它之外(如果你遇到任何类型的运行时异常,你会窃取连接)? @niraj - 我们可以写 Statement.RETURN_GENERATED_KEYS 而不是 ps.RETURN_GENERATED_KEYS 因为它是 java.sql.Statement 类中的静态变量。【参考方案4】:您可以使用 prepareStatement
方法获取额外的 int
参数
PreparedStatement ps = con.prepareStatement(sql, Statement.RETURN_GENERATED_KEYS)
对于某些 JDBC 驱动程序(例如 Oracle),您必须明确列出生成键的列名或索引:
PreparedStatement ps = con.prepareStatement(sql, new String[]"USER_ID")
【讨论】:
我接受了您的回答,因为您展示了更多实现相同结果的方法。【参考方案5】:我现在没有编译器,我会通过提问来回答:
你试过了吗?有用吗?
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
的子类....查看我更新的帖子。以上是关于带有 Statement.RETURN_GENERATED_KEYS 的 PreparedStatement的主要内容,如果未能解决你的问题,请参考以下文章
如何翻转正面带有标签而背面带有另一个标签的视图 - 参见图片
CakePHP 如何处理带有/不带有 'id' 字段的 HABTM 表?
带有 RecyclerView 的 DialogFragment 比带有 Recyclerview 的 Fragment 慢