如何理解T-SQL中Merge语句(二)

写在前面的话:上一篇写了如何理解T-SQL中Merge语句,基本把Merge语句要讲的给讲了,在文章的后面,抛出了几个结,当时没有想明白怎么去用文字表达,这一篇就来解答一下这几个结,又是一篇“天马行空”的文字,大家凑合看吧。

===正文开始===

  先看下面表一(Student_Target)和表二(Student_Source)。

一、When Matched部分  

  执行下面SQL语句:

MERGE INTO Student_Target AS st
USING Student_Source AS ss
ON st.Sno = ss.Sno
    WHEN MATCHED
        THEN UPDATE SET st.Sname = ss.Sname
;

  执行完上面SQL语句后,现在的Student_Target表的内容应该也很容易得到,如下图:

  Student_Target表中Sno=1,2的行的Sname值被Student_Source中相应的值update,不过这个结果是怎么来的呢?今天用另外一种思路来理解一下怎么得到的,我们增加一个中间过程,表述如下:

  (1)上面的SQL语句:MERGE INTO......WHEN MATCHED,可以类比成一个inner join语句:select * from Student_Target as st inner join Student_Source as ss on st.Sno=ss.Sno,内部联接后,结果如下图:

  其中红色框内是Student_Target部分,蓝色为Student_Source部分,这个图表示的是什么呢?我们可以认为后面执行的操作仅仅影响图中红色的Student_Target部分,因此,执行Then update set st.sname=ss.sname,那么原始表一Student_Target中只有上图红色框中的部分受影响,即红框中Sname依次被蓝框内Sname给update了,其他行(3,‘cc‘)不受影响,因此Student_Target最后结果为:

  (2)照此思路,如果Then update set st.sname=ss.sname改成Then delete,自然只是红色框中部分被删除,最后Student_Target结果如下图,仅仅留下了一行。

  (3)照此思路,如果Then update set st.sname=ss.sname改成Then insert values(ss.sno,ss.sname),会出现什么情况呢?再看前面的红色和蓝色部分,红色部分内容不为NULL,因此无法用右边的蓝色部分给insert进去,所以应该报错,执行一下,果然报下面的错误:

   

  现在应该明白When Matched为什么不允许有insert语句了吧,那么什么时候允许insert语句呢,接着往下看。

二、When NOT Matched BY Target部分  

  前面When Matched我们通过inner join的思路进行了理解,这次When NOT Matched BY Target我们用right outer join的思路去想想,同样原始表格还是表一和表二,再次贴图如下:

 

执行下面SQL语句:

MERGE INTO Student_Target AS st
USING Student_Source AS ss
ON st.Sno = ss.Sno
    WHEN NOT MATCHED BY TARGET
        THEN INSERT VALUES(ss.Sno,ss.Sname)
;

  先不用着急看上面运行结果,我们先用right outer join的思路去梳理一下:

  (1)上面的SQL语句:MERGE INTO......WHEN NOT MATCHED BY TARGET,可以类比成一个right outer join语句:select * from Student_Target as st right outer join Student_Source as ss on st.Sno=ss.Sno,右外部联接后(你应该对right outer join 概念非常清晰吧),结果如下图:

  其中红色框内是Student_Target部分,蓝色为Student_Source部分,这个图表示的是什么呢?我们可以认为后面执行的操作仅仅影响图中红色的Student_Target部分,因此,执行Then insert values(ss.Sno,ss.Sname),那么原始表一Student_Target中只有上图红色框中的部分受影响,即红框中Sno和Sname为null的依次被蓝框内Sno和Sname给insert了,这里之所以可以insert,就是因为第三行和第四行里面有null的值,因此最后Student_Target的运行结果为:

  (2)照此思路,如果Then insert values(ss.Sno,ss.Sname)改成Then delete,自然只是红色框中部分被删除,但是红色部分的第三行和第四行是全为NULL的值,全null了自然无法删除啊,因此应该报错,运行一下,果然报错如下:

  (3)照此思路,如果Then insert values(ss.Sno,ss.Sname)改成Then update set st.Sname=ss.Sname,同样因为全null的行会报错,因为全null无法更新啊,报错信息如下:

   

  现在应该明白When NOT Matched BY Target为什么不允许有update和delete语句了吧。

三、When NOT Matched BY SOURCE部分  

  前面When Matched我们通过inner join的思路进行了理解,When NOT Matched BY Target用right outer join的思路去理解了,这次的When not matched by source自然应该用left outer join 来理解了,具体理解过程仿照第二部分,就不详细写了,直接上结果吧:

  同样原始表格还是表一和表二,贴图如下:

  (1)执行如下SQL语句:select * from Student_Target as st left outer join Student_Source as ss on st.Sno=ss.Sno,左外部联接后,结果如下图:

 

  (2)执行下面SQL语句:

MERGE INTO Student_Target AS st
USING Student_Source AS ss
ON st.Sno = ss.Sno
    WHEN NOT MATCHED BY SOURCE
        THEN delete
;

  最后Student_Target的运行结果为:

  删除了Student_Target中有而Student_Source中没有的行。

  (2)执行insert和update语句同样报错。

四、最后总结

  很佩服你竟然能看到这里,最后来一个总结吧:

(1)when matched:类比inner join去思考,可以执行update和delete操作,无法执行insert操作。

(2)when not matched by target:类比right outer join去思考,可以执行insert操作,无法执行update和delete操作。(target表中没有(not matched),而source表中有)

(3)when not matched by source,类比left outer join去思考,可以执行delete 操作,无法执行insert和update操作。(target表中有,而source表中没有(not matched))

备注:练习过程中用到的代码:

USE testDB
GO
MERGE INTO Student_Target AS st
USING Student_Source AS ss
ON st.Sno = ss.Sno
    --WHEN MATCHED
        --THEN UPDATE SET st.Sname = ss.Sname
        --THEN  INSERT VALUES(ss.sno,ss.sname)
    --WHEN NOT MATCHED BY TARGET
		--THEN UPDATE set st.sname=ss.sname
    --WHEN NOT MATCHED BY SOURCE
		--THEN UPDATE set st.Sname=ss.Sname,st.Sno=ss.Sno

;

TRUNCATE TABLE dbo.Student_Target

TRUNCATE TABLE dbo.Student_Source

insert into Student_Target (sno,sname)values(1,‘aa‘),(2,‘bb‘),(3,‘cc‘)

insert into Student_Source (sno,sname)values(1,‘xiaoming‘),(2,‘xiaoli‘),(4,‘xiaohong‘),(5,‘xiaoping‘)

SELECT * FROM dbo.Student_Target

SELECT * FROM dbo.Student_Source

SELECT * FROM dbo.Student_Target AS st INNER JOIN dbo.Student_Source AS ss
ON st.Sno=ss.Sno

SELECT * FROM dbo.Student_Target AS st RIGHT OUTER JOIN dbo.Student_Source AS ss
ON st.Sno=ss.Sno

SELECT * FROM dbo.Student_Target AS st LEFT OUTER JOIN dbo.Student_Source AS ss
ON st.Sno=ss.Sno
时间: 2024-10-15 00:57:26

如何理解T-SQL中Merge语句(二)的相关文章

SQL中常见语句

SQL中常见语句笔记: --替换字段中的回车符和换行符 UPDATE [dbo].[MGoods_Test] SET GoodsName = REPLACE(GoodsName, CHAR(13) + CHAR(10), '') --删除表命令 DROP TABLE [dbo].[MGoods_Test] --删除表中数据命令 DELETE FROM [dbo].[MGoods_Test] --逐行删除 有日志记录 TRUNCATE TABLE [dbo].[MGoods_Test] --TRU

如何理解T-SQL中Merge语句

写在前面的话:之前看过Merge语句,感觉没什么用,完全可以用其他的方式来替代,最近又看了看Merge语句,确实挺好用,可以少写很多代码,看起来也很紧凑,当然也有别的优点. ====正文开始===== SQL Server 2008 引入了Merge关键字,主要是在一条语句里面可以执行insert.update.delete操作,以实现用一个源对象的数据对目标对象数据进行操作.注意这里的”源对象“和”目标对象“我用黑色标注了,源对象和目标对象实际上不仅仅可以是表Table,还可以是临时表.视图.

Oracle中MERGE语句的使用

Oracle在9i引入了merge命令, 通过这个merge你能够在一个SQL语句中对一个表同时执行inserts和updates操作. 当然是update还是insert是依据于你的指定的条件判断的,Merge into可以实现用B表来更新A表数据,如果A表中没有,则把B表的数据插入A表. MERGE命令从一个或多个数据源中选择行来updating或inserting到一个或多个表 语法如下 MERGE INTO [your table-name] [rename your table her

Oracle 中MERGE语句的用法(转载)

原文章出处(http://blog.csdn.net/lichkui/article/details/4306299) MERGE语句是Oracle9i新增的语法,用来合并UPDATE和INSERT语句.通过MERGE语句,根据一张表或子查询的连接条件对另外一张表进行查询,连接条件匹配上的进行UPDATE,无法匹配的执行INSERT.这个语法仅需要一次全表扫描就完成了全部工作,执行效率要高于INSERT+UPDATE. //表1create table subs(msid number(9), 

SQL Server Merge语句的使用

Merge关键字在SQL Server 2008被引入,它能将Insert,Update,Delete简单的并为一句.MSDN对于Merge的解释非常的短小精悍:”根据与源表联接的结果,对目标表执行插入.更新或删除操作.例如,根据在另一个表中找到的差异在一个表中插入.更新或删除行,可以对两个表进行同步.”,通过这个描述,我们可以看出Merge是关于对于两个表之间的数据进行操作的. 可以想象出,需要使用Merge的场景比如: 数据同步 数据转换 基于源表对目标表做Insert,Update,Del

SQL中游标(二)

游标是SQL数据库中不可或缺的部分,可以旋转储存在系统永久表中的数据行的副本,下面就将为您详解游标的使用,以及语法,供您参考学习. MS-SQL的游标是一种临时的数据库对象,既对可用来旋转储存在系统永久表中的数据行的副本,也可以指向储存在系统永久表中的数据行的指针. 游标为您提供了在逐行的基础上而不是一次处理整个结果集为基础的操作表中数据的方法. 1.如何使用游标 1)    定义游标语句 Declare <游标名> Cursor For 2)    创建游标语句 Open <游标名&g

使用正则表达式实现像SQL中LIKE语句中的%和_通配

在项目中我们经常遇到将数据库的数据取到后再次进行筛选过滤的情况.LINQ to Entity提供了统一的查询接口并且可以高效的完成工作,但是对于我们常在SQL中使用的%和_这样的通配符并没有支持.我们只能通过String.Contains方法来实现简单的通配.使用String.Contains方法是无法达到在查询串中使用通配符的目的的.正则表达式虽然晦涩难懂,但功能十分强大,解决个统配符绰绰有余. 代码如下:     public static class LINQHelper     {   

ORA-03113 SQL中select语句全表扫描带来的异常

今天在ERP系统的维护过程中,业务人员反馈了一个问题过来,是ERP系统生产单模块的预览打印报表出错,看到后我逐步做了以下的排查: 1.尝试其他单据是否存在相同问题 2.直接打开水晶报表,将参数代入看看是否是报表问题 排查之后逐渐发现,问题出在数据源身上,找到返回数据集的存储过程,进入测试窗口检查是否运行正常,结果发现运行即进入卡死状态,进程无法中断,只好强行退出PL/SQL,这时候我估计到问题出在SQL语句上,因此将SQL语句复制到新的窗口,代入参数,如下: SELECT WO_NBR,WO_L

SQL 中 SELECT 语句的执行顺序

好像自已在书写 SQL 语句时由于不清楚各个关键字的执行顺序, 往往组织的 SQL 语句缺少很好的逻辑, 凭感觉 "拼凑" ( 不好意思, 如果您的 SQL 语句也经常 "拼凑", 那您是不是得好好反省一下呢?, 呵呵). 确实是爽了自己, 可苦了机器, 服务器还需要在我们的杂乱无章的 SQL 语句中寻找它下一句需要执行的关键字在哪里. 效率嘛, 由于我们的感觉神经对秒以下的变化实在不敏感, 暂且就认为自已写的 SQL 顺序无关紧要, "反正没什么变化!&