您的位置:首页 > 其它

DB2的MERGE语句使用注意事项!

2012-07-24 17:39 405 查看

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

ELSE IGNORE
个人感觉ELSE IGNORE就是个废物,写跟不写好像没有区别,运行结果都一样。

关于UPDATE,DELETE/INSERT
对于第二条与第三条规则,有个方便的记忆方法。当MATCHED的时候,即有满足条件的数据时,就可以对这些数据进行UPDATE或DELETE操作,如果NOT MATCHED,则INSERT新的数据。简单地说,有数据就更新或删除,无数据就插入。

例外的数据
用MERGE语句INSERT的数据不能被MERGE的分支语句所操作。

UPDATE的赋值语法
UPDATE的有两种可用的赋值语法,一种是“column = ?[, column = ?] ...”,另一种是“(column, column ...) = (value, value ...)”。

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

上面三个例子中的ELSE IGNORE都可以忽略,并且不会影响最终结果。

官方文档参考:http://publib.boulder.ibm.com/infocenter/db2luw/v9/index.jsp?topic=/com.ibm.db2.udb.admin.doc/doc/r0010873.htm
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: