使用 MERGE 语句实现增删改
Ø 简介
在平常编写增删改的 SQL 语句时,我们用的最多的就是 INSERT、UPDATE 和 DELETE 语句,这是最基本的增删改语句。其实,SQL Server 中还有另外一个可以实现增删改的语句,那就是本文将要介绍的 MERGE 语句。
1. 什么是 MERGE 语句
1) MERGE 语句可以在一条语句中,根据与源表联接的结果,对目标表执行插入、更新或删除操作。 例如,根据在另一个表中找到的差异在一个表中插入、更新或删除行,可以对两个表进行同步。
2) 为什么使用 MERGE 语句呢?
1. 试想一下,如果我们想要完成这件事情:将查询的结果集写入 A 表中,如果 A 表已经存在了该记录,就执行 UPDATE 语句(更新);否则,执行 INSERT 语句(新增)。(删除也是如此)
2. 如果按照我们的思路,是不是首先应该查出数据,在根据关键字段去 A 表中匹配,如果匹配到则执行 UPDATE 语句,否则执行 INSERT 语句。但是这样一来我们必须执行两次查询结果集:第一次查询比对,第二次查询后执行新增或更新操作,这样就相当繁琐也影响性能。
3. 而 MERGE 语句可以很轻松的帮我们解决这个问题,帮我们简洁了代码和优化性能。
2. 我的示例
1) 使用 MERGE语句完成新增或更新操作
INSERT INTO Sys_DbLog(LogTime, Type, Itme, Message, Remark) VALUES(GETDATE(), 'info', 'SP_Update_AchievCommission', '存储过程执行', NULL);
BEGIN TRY
WITH ce1(OrderNum, TotalMoney) AS
(
SELECT OrderNum, SUM(T2.TotalMoney) AS TotalMoney FROM Orders AS T1
INNER JOIN OrderDetail AS T2 ON(T1.Id = T2.OrderId)
WHERE 1=1
AND (T1.OrderStatusId > 1 AND T1.OrderStatusId < 10)
AND (T1.PayStatusId = 2 AND T1.PayTime >= @StartTime AND T1.PayTime <= @EndTime)
GROUP BY T1.OrderNum
) --订单明细的总实付金额
MERGE INTO Crm_AchievCommission AS T1
USING
(
SELECT T1.OrderNum, T1.PayTime, T1.CityId AS OrderCityId, T3.CityId AS CustCityId, T4.CityId AS EmplCityId, T4.EmployeeId, T5.ProductId,
CAST(ROUND((T1.RealTotal - ISNULL(T1.Freight, 0)) * (ISNULL(T5.TotalMoney, 0) / T7.TotalMoney), 2) AS numeric(19,2)) AS RealTotal/*四舍五入保留两位小数*/,
T6.CommissionRate, T6.ExtraCommissionRate FROM Orders AS T1
INNER JOIN UserInfo AS T2 ON(T1.UserId = T2.Id)
INNER JOIN Customer AS T3 ON(T2.CustomerId = T3.Id)
INNER JOIN Sys_EmployeeProfile AS T4 ON(T1.SalesUserId = T4.EmployeeId)
INNER JOIN OrderDetail AS T5 ON(T1.Id = T5.OrderId)
INNER JOIN Crm_CommissionRates AS T6 ON(T5.ProductId = T6.ProductId)
INNER JOIN ce1 AS T7 ON(T1.OrderNum = T7.OrderNum)
WHERE 1=1
AND T1.UserId > 0
AND T2.CustomerId > 0
AND T1.SalesUserId > 0
AND (T1.OrderStatusId > 1 AND T1.OrderStatusId < 10)
AND (T1.PayStatusId = 2 AND T1.PayTime >= @StartTime AND T1.PayTime <= @EndTime)
) AS T2 ON(T1.OrderNum = T2.OrderNum AND T1.ProductId = T2.ProductId)
WHEN MATCHED
THEN UPDATE SET
OrderCityId = T2.OrderCityId,
CustCityId = T2.CustCityId,
EmplCityId = T2.EmplCityId,
EmployeeId = T2.EmployeeId,
RealTotal = T2.RealTotal,
UpdateTime = GETDATE()
WHEN NOT MATCHED
THEN INSERT(OrderNum, PayTime, OrderCityId, CustCityId, EmplCityId, EmployeeId, ProductId, RealTotal, CommissionRate, ExtraCommissionRate, CreateTime, UpdateTime)
VALUES(T2.OrderNum, T2.PayTime, T2.OrderCityId, T2.CustCityId, T2.EmplCityId, T2.EmployeeId, T2.ProductId, T2.RealTotal, T2.CommissionRate, T2.ExtraCommissionRate, GETDATE(), NULL);
END TRY
BEGIN CATCH
DECLARE @Message nvarchar(4000) = '错误行号:' + CAST(ERROR_LINE() AS nvarchar(20)) + ',错误信息:' + ERROR_MESSAGE();
INSERT INTO Sys_DbLog(LogTime, Type, Itme, Message, Remark) VALUES(GETDATE(), 'error', 'SP_Update_AchievCommission', @Message, NULL);
END CATCH
3. 更多介绍
1) 官网介绍
https://docs.microsoft.com/zh-cn/sql/t-sql/statements/merge-transact-sql?view=sql-server-2017
2) 其他
https://www.cnblogs.com/billqian/p/5157052.html
使用 MERGE 语句实现增删改的更多相关文章
- 四种简单的sql语句(增删改查语句)
四种简单的sql语句(增删改查语句) 一.插入语句 insert into [table] ([column],[column],[column]) values(?,?,?) 二.删除语句 dele ...
- dml语句就是你常写的sql语句,增删改查
dml语句就是你常写的sql语句,增删改查
- 利用sql语句进行增删改查
1.查询 函数:raw(sql语句) 语法:Entry.objects.raw(sql) 返回:QuerySet 2.增删改 from django.db import connection def ...
- 常见 SQL语句使用 增删改查
一.常见的增删改查(一).查:1.SELECT 列名称 FROM 表名称,其中列名可以是多个,中间用豆号分开,如SELECT LastName,FirstName FROM Persons: 2.SE ...
- SQL简单语句(增删改查)
简单的SQL语句增删改查操作 说明: 在mysql里面亲测结果正确 用到的表(学生表:studnets) 1.创建一个学生表,(学号,姓名,性别,家庭住址) mysql> create t ...
- SQL语句简单增删改查
常用数据类型 Int:整数,长度没有作用 Varchar:字符串,varchar(3)表示最多存放3个字符,1个中文或英文或符合都占1个字符 Decimal:小数,decimal(5,2)表示共存5位 ...
- SQL SQL语句的增删改查
一.增:有2种方法 1.使用insert插入单行数据: 语法:insert [into] <表名> [列名] values <列值> 例:insert into Strdent ...
- SQL语句的增删改查
一.增:有2种方法 1.使用insert插入单行数据: 语法:insert [into] <表名> [列名] values <列值> 例:insert into Strdent ...
- SQL语句的增删改查(详细)--转载
转载源: http://blog.csdn.net/a88055517/article/details/6736284/ 一.增:有2种方法 1.使用insert插入单行数据: 语法:insert [ ...
随机推荐
- 模拟vue的tag属性,在react里实现自定义Link
我封装了一个简单的实现react里自定义Link的方法,方便大家使用. 因为普通组件没有metch.location.history等属性.只有在<Router>里面的<compon ...
- ✔ OI Diary ★
一 | 2019-3-28 1.整晨,云之考矣,暴后皆不会,邃无感而写斯普雷尔,然则午后知暴可六十哉. 然则斯普雷毙,虽特判之矣,然则暴只判二十哉,呜呼! 2.午间归宿,视白购书一本,目触,感之甚集 ...
- TPYBoard开发板搭建,实现隐秘通信
一.准备工作 lTPYBoard v102(简称v102) 1块 lTPYBoard v202(简称v202) 1块 l杜邦线.MicroUSB数据线 若干 (成本100元以内,某宝上可以买到) 附上 ...
- 排序学习实践---ranknet方法
要: 1 背景 随着移动互联网的崛起,越来越多的用户开始习惯于从手机完成吃.喝.玩.乐.衣.食.住.行等各个方面的需求.打开手机,点开手淘.美团等APP,商品玲玲满目,而让用户将所有商品一页 ...
- GEC6818连接Ubuntu,下载程序至开发板
使用 secure CRT连接开发板,可视化操作 连接成功 设置临时ip ubuntu 要跟 开发板同一网段: ip前三位相同 代码:sudo service tftpd-hpa restart 代 ...
- hMailServer相关视频教程
来源:https://www.hmailserver.org/viewtopic.php?f=4&t=34
- JS-JAVASCRIPT的eval()方法
Lodop的传统模版是JS语句,如果储存的这种JS模版,可用javascript的eval()方法获取,该方法属于JS的范畴,并不是lodop语句,由于这个方法很多程序员可能不常用,对于这个JS方法不 ...
- Python学习之路——迭代器
迭代器 # 通过迭代器取值优缺点: # 优点:不依赖索引,完成取值 # 缺点:不能计算长度,不能指定位取值(只能从前往后逐一取值) 可迭代对象 ''' 可迭代对象: 有__iter__()方法的对象, ...
- vue 使用小结 2019.03
v-bind 中使用函数 :attr = 'num' 如上面的例子,通常 num 是 vue 实例中 data 的值,或者是 computed 对象中的值,我们可以在具体函数中计算,改变相应的变量,以 ...
- java 将保单数据 生成图片
主要代码:---------------------------------------------------------------- /** * 生成图片 * @param cellsValue ...