7

我试图在事务范围内获取插入行的主键,因为我不想让数据库处于逻辑上不一致的状态。

我的问题是我找不到一种方法来检索以前执行的查询的 ID 值,我想将其用于下一个插入查询。在事务生效时查询 PostgreSQL 数据库在非外键表中显示没有结果(该行尚未提交?)。我相信这是由于事务的隔离级别。

下面是我试图用生产代码做的事情,尽管为了清楚起见,稍微编辑并缩小到一个功能。const int lastInsertId始终为 0,在这种情况下应该意味着没有找到任何值(从技术上讲,该toInt()函数失败了)。我尝试手动插入一个有效的非外键行,然后调用LASTVAL()它产生了预期的结果——插入行的 ID。

那么,我做错了什么?我在这里遗漏或误解了什么?

void createEntityWithoutForiegnKeyConstraint(const QString &nameOfEntity)
{
  db_.transaction();

  QSqlQuery insertQuery(db_);
  insertQuery.prepare("INSERT INTO \"EntityWithoutForeignKey\" (\"name\") VALUES (:name);");

  insertQuery.bindValue(":name", nameOfEntity);
  execQuery(__LINE__, insertQuery);
  QSqlQuery lastIdQuery("SELECT LASTVAL();", db_); // auto executes
  const int lastInsertId = lastIdQuery.value(0).toInt();

  if (lastInsertId <= 0) // 0 is not a valid ID
    throw exception("Oh noes.");

  createEntityWithForeignKeyConstraint(lastInsertId, someData);

  if (!db_.commit())
    db_.rollback();
}
4

2 回答 2

16

我意识到这是一个老问题,但在 Qt 5.10(可能更早)中有一个QSqlQuery::lastInsertId()可以在QSqlQuery::exec().

如果您使用的数据库(例如 SQLite)不支持语句中的RETURNING子句,这将非常有用INSERT

QSqlQuery::lastInsertId() 文档。

用法大致如下:

QSqlQuery q;
q.prepare("INSERT INTO table_name VALUES(:some_column_name)");
q.bindValue(":some_column_name", "FooBar");
q.exec();

qDebug() << "Last ID was:" << q.lastInsertId();
于 2018-01-14T20:37:12.837 回答
2

@Kasheen 完全正确,但我想将重点放在您应该牢记的一个重要方面:不要忘记将所有内容封装在一个事务中。

为什么?如果使用生成的主键(你得到了q.lastInsertId())的第二次插入失败,它可以节省时间并避免数据库损坏。

所以你的插入应该是这样的(这基本上是@Kasheen 的答案,有一些补充):

db_.transaction(); // Starts a transaction

QSqlQuery q;

// first insert
q.prepare("INSERT INTO table_name VALUES(:some_column_name)");
q.bindValue(":some_column_name", "FooBar");
q.exec();

auto pk == q.lastInsertId().toInt;

// second insert
q.prepare("INSERT INTO other_table VALUES(:other_column_name, :fk)");
q.bindValue(":other_column_name", "OtherText");
q.bindValue(":fk", pk);
q.exec();

db_.commit(); // Commits transaction

@average joe 正确地进行了事务处理,但如果您只查看解决方案,您可能会忘记它。

于 2018-09-15T09:44:38.530 回答