SQL标准UPSERT调用

Bob*_*der 8 mysql sql upsert hsqldb

我正在寻找一个标准的SQL" UPSERT"语句.如果存在,则插入和更新一次调用.

我正在寻找一个有效,高效的跨平台电话.

我见过MERGE,UPSERT,REPLACE,INSERT .. ON DUPLICATE UPDATE但没有声明满足需求.

顺便说一句,我使用MYSQL和HSQLDB进行unitests.我知道HSQLDB是有限的,可能无法满足我的需求,但即使没有它,我也找不到标准的方法.现在只有MYSQL和HSQLDB才足够的声明.

我一直在寻找一段时间,但无法得到答案.

我的桌子:

CREATE TABLE MY_TABLE (
  MY_KEY varchar(50) NOT NULL ,
  MY_VALUE varchar(50) DEFAULT NULL,
  TIME_STAMP bigint NOT NULL,
  PRIMARY KEY (MY_KEY)
);
Run Code Online (Sandbox Code Playgroud)

任何的想法?

Ike*_*ker 6

在单个命令中执行upsert的语法因RDBMS而异.

请参阅Wikipedia了解更多信息.

如果您需要跨平台解决方案,那么您将需要使用多个命令.首先检查现有行,然后根据需要有条件地插入或更新.


Bil*_*win 6

MySQL和HSQLDB支持的唯一解决方案是查询要替换的行,并有条件地查询INSERT或UPDATE.这意味着您必须编写更多应用程序代码来补偿RDBMS实现之间的差异.

  1. 开始交易.
  2. SELECT ... FOR UPDATE.
  3. 如果SELECT找到行,则UPDATE.
  4. 另外,INSERT.
  5. 承诺.

MySQL不支持ANSI SQL MERGE语句.它支持REPLACE和INSERT ... ON DUPLICATE KEY UPDATE.有关详细信息,请参阅我对"INSERT IGNORE"和"INSERT ... ON DUPLICATE KEY UPDATE"的回答.


评论:是的,另一种方法是尝试INSERT并查看它是否成功.否则,请进行更新.如果您尝试INSERT并且它遇到重复键,它将生成错误,这在某些客户端接口中变为异常.在MySQL中执行此操作的缺点是,即使INSERT失败,它也会生成新的自动增量ID.所以你最终会有差距.我知道自动增量序列中的差距通常不值得担心,但是我去年帮助了一位客户,由于这种影响,成功插入之间的差距为1000-1500,结果是他们耗尽了范围. INT在他们的主键中.

正如@baraky所说,首先可以尝试更新UPDATE,如果这会影响零行,则执行INSERT.我对此策略的评论是,更新零行也不例外 - 您必须在UPDATE之后检查"受影响的行数"以了解它是否"成功".

但是查询受影响的行数会使您回到最初的问题:您必须在MySQL和HSQLDB中使用不同的查询.

HSQLDB:

CALL DIAGNOSTICS(ROW_COUNT);
Run Code Online (Sandbox Code Playgroud)

MySQL的:

SELECT ROW_COUNT();
Run Code Online (Sandbox Code Playgroud)