MERGE INTO project px
USING (SELECT project.* FROM SYSIBM.DUAL LEFT JOIN project WHERE id = ?) py
ON px.id = py.id
WHEN MATCHED THEN
UPDATE SET
(name, description, last_changed_time) = (?, ?, ?)
WHEN NOT MATCHED THEN
INSERT (name, description, last_changed_time)
VALUES (?, ?, ?)
ELSE IGNORE;
SELECT IDENTITY_VAL_LOCAL() FROM SYSIBM.DUAL 函数IDENTITY_VAL_LOCAL对于MERGE语句中的INSERT,不能像普通的INSERT语句那样返回最后一次插入数据库中的数据的ID值,对于这种情况,只能使用普通的INSERT语句。
WHEN MATCHED THEN MATCHED的情况下,只有UPDATE和DELETE语句可以使用,不能使用INSERT语句。
WHEN NOTE MATCHED THEN NOTE MATCHED的情况下,不能使用UPDATE和DELETE语句,只可以使用INSERT语句。
WHEN [NOT] MATCHED AND ( ... AND ... OR ... ) THEN 除了使用默认的MATCHED/NOT MATCHED以外,还可以指定额外的判断条件,但与MATCHED并列的条件只能用AND,而AND里面可以使用AND和OR。
空结果集的影响 在上面的例子中,使用了伪表左外连接project表,这样做的目的是保证查询结果至少有一条数据,不管project表中是否有数据存在。假如不使用DUAL而直接使用project作为查询目标表的话,有可能会得到零记录的结果集,警告编号IWAQ0003W,这个时候ON子句中的比较就可能会无任何意义,因为NULL与任何值比较的结果既不是TRUE也不是FALSE,结果永远都是NULL。
警告内容:No row was found for FETCH, UPDATE or DELETE; or the result of a query is an empty table.. SQLCODE=100, SQLSTATE=02000
SIGNAL 就是发出一个错误信号,终止当前语句。从使用上说可能比较方便,但设计上讲感觉有点问题,目前还没见过可以在SQL语句运行这样的STATEMENT,有点像存储程序里的功能,抛出异常,实在是奇怪,不伦不类的NATIVE FEATURE。
关于SQLSTATE的官方解释:【SQLSTATE classes that begin with the characters '7' through '9' or 'I' through 'Z' may be defined. Within these classes, any subclass may be defined. SQLSTATE classes that begin with the characters '0' through '6' or 'A' through 'H' are reserved for the database manager. Within these classes, subclasses that begin with the characters '0' through 'H' are reserved for the database manager. Subclasses that begin with the characters 'I' through 'Z' may be defined.】。在使用的时候,一般使用数字7-9或字母I-Z开头来定义我们自己的SQLSTATE,长度为5。
一些例子:
MERGE INTO project px
USING (SELECT 1 AS VAL FROM SYSIBM.DUAL WHERE 1 = 2) py
ON VAL = 1
WHEN MATCHED THEN
SIGNAL SQLSTATE 'ZZ110'
SET MESSAGE_TEXT = 'Kidding you?'
WHEN NOT MATCHED THEN
SIGNAL SQLSTATE 'ZZ911'
SET MESSAGE_TEXT = 'Kidding me?'
ELSE IGNORE;
-- 结果是触发警告,不走MATCHED,也不走NOT MATCHED
MERGE INTO project px
USING (SELECT 1 AS VAL FROM SYSIBM.DUAL WHERE 1 = 1) py
ON VAL = 1
WHEN MATCHED THEN
SIGNAL SQLSTATE 'ZZ110'
SET MESSAGE_TEXT = 'Kidding you?'
WHEN NOT MATCHED THEN
SIGNAL SQLSTATE 'ZZ911'
SET MESSAGE_TEXT = 'Kidding me?'
ELSE IGNORE;
-- 结果走MATCHED
-- Application raised error or warning with diagnostic text: "Kidding you?".. SQLCODE=-438, SQLSTATE=ZZ911
MERGE INTO project px
USING (SELECT 1 AS VAL FROM SYSIBM.DUAL WHERE 1 = 1) py
ON VAL = 2
WHEN MATCHED THEN
SIGNAL SQLSTATE 'ZZ110'
SET MESSAGE_TEXT = 'Kidding you?'
WHEN NOT MATCHED THEN
SIGNAL SQLSTATE 'ZZ911'
SET MESSAGE_TEXT = 'Kidding me?'
ELSE IGNORE;
-- 结果走NOT MATCHED
-- Application raised error or warning with diagnostic text: "Kidding me?".. SQLCODE=-438, SQLSTATE=ZZ110