写了一个常规性生成merge 的小脚本
现在使用数据库来写存储过程,动不动参数就会用到xml ,当然罗,优势也很明显,参数相对固定,而且灵活,如果要修改或者什么的,中间接口层也不需要做变化,只需要修改封装的存储过程以及程序传参就ok了。
随着时间慢慢过,有时候就有一个存储过程,一个xml 来应对整个表的新增,修改,删除的情况了。而对于这个情况,我个人比较喜欢使用 Merge关键字来处理。但是如果表里面的列很多,那么复制黏贴啊之类的机械动作就会很多,而且没有什么价值。所以我就写了一个小脚本,应对了使用xml 来做表的增删改的作用
首先我先创建一个表
CREATE TABLE employee(
ID INT IDENTITY(1,1) PRIMARY KEY,
name NVARCHAR(50),
age INT,
birthdate DATE,
salary MONEY
)
然后我准备使用这个xml 来进行对应写入
DECLARE @employee XML='
<root>
<employee Action="1"> <!--这个Action 代表动作,1 新增 2 修改 3 删除 这样来控制比较灵活,不需要每次都一大段-->
<name>AAA</name>
<age>27</age>
<birthdate>1989-01-02</birthdate>
<salary>1200</salary>
</employee>
<employee Action="1">
<name>BBB</name>
<age>23</age>
<birthdate>1994-01-02</birthdate>
<salary>2200</salary>
</employee>
</root>
'
然后是生成的脚本。通常解析xml 会有2种的解析方法,一种是直接用openxml 来进行解析,一种是使用 xml.nodes 的函数进行取值,这里我两种都可以进行一个简单处理生成
DECLARE @TableName VARCHAR(50) = 'employee',
@XMLType TINYINT = 1, --1 使用with 格式, 2 使用nodes 格式
@Path NVARCHAR(max) = 'root/employee',
@HasAction BIT = 1 --0 没有动作 1 包含动作 DECLARE @Columns NVARCHAR(MAX), --通用列的串
@FilterColumns NVARCHAR(max), --过滤外键,主键的列
@On NVARCHAR(100), --自动生成主键去匹配
@Sql NVARCHAR(MAX) SELECT @Columns = STUFF((
SELECT ',' + name
FROM sys.columns
WHERE object_id = OBJECT_ID(@TableName)
ORDER BY column_id
FOR XML PATH('')),1,1,''),
@FilterColumns = STUFF((
SELECT ',' + name
FROM sys.columns
WHERE object_id = OBJECT_ID(@TableName)
AND is_computed = 0
AND is_identity = 0
ORDER BY column_id
FOR XML PATH('')),1,1,''),
@On = STUFF((
SELECT 'AND TAR.' + c.name + ' = SOUR.' + c.name
FROM sys.indexes a
INNER JOIN sys.index_columns b ON a.object_id = b.object_id
INNER JOIN sys.columns c ON c.object_id = b.object_id AND b.column_id = c.column_id
WHERE a.object_id = OBJECT_ID(@TableName)
AND a.is_primary_key = 1),1,4,'') SELECT @Sql = ';WITH SOUR AS(
SELECT '+ CASE WHEN @XMLType = 1 THEN REPLACE(@Columns ,',',CHAR(10) + REPLICATE(CHAR(9),2) + ',')
WHEN @XMLType = 2 THEN STUFF((SELECT CHAR(10) + ',t.c.value(''(' + a.name + '/text())[1]'',''' + TYPE_NAME(user_type_id) + CASE WHEN a.system_type_id IN (167,175) THEN '(' + CASE WHEN a.max_length = -1 THEN 'max' ELSE RTRIM(a.max_length) END+ ')'
WHEN a.system_type_id IN (231,239) THEN '(' + CASE WHEN a.max_length = -1 THEN 'max' ELSE RTRIM(a.max_length/2) END + ')'
WHEN a.system_type_id IN (59,106,108) THEN '(' + RTRIM(a.max_length) + ',' + RTRIM(a.scale) + ')'
ELSE ''
END + ''') AS ' + a.name
FROM sys.columns a
WHERE object_id = OBJECT_ID(@TableName)
ORDER BY column_id
FOR XML PATH('')),1,2,'')
ELSE '' END
+ CASE WHEN @XMLType = 1 AND @HasAction = 1 THEN CHAR(10)+ REPLICATE(CHAR(9),2) + ',[Action]'
WHEN @XMLType = 2 AND @HasAction = 1 THEN CHAR(10)+ REPLICATE(CHAR(9),2) + ',t.c.value(''@Action'',''tinyint'') [Action]'
ELSE '' END
+ '
FROM ' + CASE @XMLType WHEN 1 THEN ' OPENXML(@XmlInt,''' + @Path + ''',3)
WITH(' + STUFF((SELECT CHAR(10)+ REPLICATE(CHAR(9),6) + ',' + a.name + ' ' + UPPER(b.name) + CASE WHEN a.system_type_id IN (167,175,231,239,108) THEN '(' + CASE WHEN a.max_length = -1 THEN 'MAX' ELSE RTRIM(a.max_length) END + ')'
WHEN a.system_type_id IN (59,106,108) THEN '(' + RTRIM(a.precision) + ',' + RTRIM(a.scale)+ ')'
ELSE '' END + ' ''' + a.name + ''''
FROM sys.columns a
INNER JOIN sys.systypes b ON a.system_type_id = b.xtype AND b.status = 0
WHERE object_id = OBJECT_ID(@TableName)
AND a.is_computed = 0
ORDER BY column_id
FOR XML PATH ('')
),1,8,'') +
+ CASE WHEN @HasAction = 1 THEN CHAR(10)+ REPLICATE(CHAR(9),6) + ',[Action] tinyint ''@Action'')' ELSE ')' END
WHEN 2 THEN ' @' + @TableName + '.nodes('''+@Path+''') as t(c)'
ELSE '' END
+ '),
TAR AS( SELECT ' + REPLACE(@Columns,',',CHAR(10) + REPLICATE(CHAR(9),2) + ',') + '
FROM ' + @TableName + ')
MERGE TAR
USING SOUR
ON '+@On+'
WHEN NOT MATCHED ' + CASE WHEN @HasAction = 1 THEN ' AND SOUR.[Action] = 1 ' ELSE '' END + '
THEN INSERT(' + @FilterColumns + ')' + CHAR(10) + REPLICATE(CHAR(9),5) + ' VALUES (SOUR.' + REPLACE(@FilterColumns,',',',SOUR.') + ')
WHEN MATCHED ' + CASE WHEN @HasAction = 1 THEN ' AND SOUR.[Action] = 2 ' ELSE '' END + ' THEN UPDATE SET '
+ STUFF(( SELECT ',' + CHAR(10) + REPLICATE(CHAR(9),5) + 'TAR.[' + name + ']= SOUR.[' + name + ']'
FROM sys.columns a
WHERE object_id = OBJECT_ID(@TableName)
AND is_computed = 0
AND is_identity = 0
AND NOT EXISTS(SELECT * FROM sys.foreign_key_columns WHERE parent_object_id = a.object_id AND parent_column_id = a.column_id)
ORDER BY column_id
FOR XML PATH('')
),1,6,'') + '
'+ CASE WHEN @HasAction = 1 THEN ' WHEN MATCHED AND SOUR.[Action] = 3 ' ELSE
'WHEN MATCHED BY SOURCE ' END + ' THEN Delete;'
PRINT @Sql
(因为偷懒,所以使用的openxml 里面的那个 sp_xml_preparedocument 这里我是没有写的)(*^__^*) 嘻嘻……
然后看下生成的情况,这个是使用xml.nodes 来生成的
;WITH SOUR AS(
SELECT t.c.value('(ID/text())[1]','int') AS ID
,t.c.value('(name/text())[1]','nvarchar(50)') AS name
,t.c.value('(age/text())[1]','int') AS age
,t.c.value('(birthdate/text())[1]','date') AS birthdate
,t.c.value('(salary/text())[1]','money') AS salary
,t.c.value('@Action','tinyint') [Action]
FROM @employee.nodes('root/employee') as t(c)),
TAR AS( SELECT ID
,name
,age
,birthdate
,salary
FROM employee)
MERGE TAR
USING SOUR
ON TAR.ID = SOUR.ID
WHEN NOT MATCHED AND SOUR.[Action] = 1
THEN INSERT(name,age,birthdate,salary)
VALUES (SOUR.name,SOUR.age,SOUR.birthdate,SOUR.salary)
WHEN MATCHED AND SOUR.[Action] = 2 THEN UPDATE SET TAR.[name]= SOUR.[name],
TAR.[age]= SOUR.[age],
TAR.[birthdate]= SOUR.[birthdate],
TAR.[salary]= SOUR.[salary]
WHEN MATCHED AND SOUR.[Action] = 3 THEN Delete;
xml.nodes
这个是使用openxml来生成的
;WITH SOUR AS(
SELECT ID
,name
,age
,birthdate
,salary
,[Action]
FROM OPENXML(@XmlInt,'root/employee',3)
WITH(ID INT 'ID'
,name NVARCHAR(100) 'name'
,age INT 'age'
,birthdate DATE 'birthdate'
,salary MONEY 'salary'
,[Action] tinyint '@Action')),
TAR AS( SELECT ID
,name
,age
,birthdate
,salary
FROM employee)
MERGE TAR
USING SOUR
ON TAR.ID = SOUR.ID
WHEN NOT MATCHED AND SOUR.[Action] = 1
THEN INSERT(name,age,birthdate,salary)
VALUES (SOUR.name,SOUR.age,SOUR.birthdate,SOUR.salary)
WHEN MATCHED AND SOUR.[Action] = 2 THEN UPDATE SET TAR.[name]= SOUR.[name],
TAR.[age]= SOUR.[age],
TAR.[birthdate]= SOUR.[birthdate],
TAR.[salary]= SOUR.[salary]
WHEN MATCHED AND SOUR.[Action] = 3 THEN Delete;
openxml
恩~然后就可放进去执行啦~~
这里只是一个很基本的用法。有几点要说明的
1、Xml的名称我默认和表名一致,有需要请改动
2、On的匹配模型我是使用主键来进行对应
其它如果有什么问题请告诉我补充~
写了一个常规性生成merge 的小脚本的更多相关文章
- 用c#写的一个局域网聊天客户端 类似小飞鸽
用c#写的一个局域网聊天客户端 类似小飞鸽 摘自: http://www.cnblogs.com/yyl8781697/archive/2012/12/07/csharp-socket-udp.htm ...
- Python 写了一个批量生成文件夹和批量重命名的工具
Python 写了一个批量生成文件夹和批量重命名的工具 目录 Python 写了一个批量生成文件夹和批量重命名的工具 演示 功能 1. 可以读取excel内容,使用excel单元格内容进行新建文件夹, ...
- 写了一个hiero中添加自定义Token的脚本
Hiero自带Token往往不够用,shotname中自带版本号的情况下要升级版本会很麻烦,比如Shot_0001_v001这样一个序列名,要升级为Shot_0001_v002就必须把_v001之前的 ...
- 写了一个web使用向导的小插件
运行效果: 引入插件: <link rel="stylesheet" href="ez-guide.css"> <script src=&qu ...
- 闲着无聊时写的一个调用天气 API 的小 Demo
分为两个部分--调用以及实现,并且由于不想折腾,直接使用了 Console 来调用. 通过firefox直接调用 Main 入口,调用以及输出 调用部分没什么好说的,主要是针对 dynamic 类型的 ...
- 用Go语言写了一个电脑搜索文件的小东西
package main import ( "bytes" "fmt" "os" "os/exec" "pat ...
- 用Python写了个下载快手视频的小脚本
最近又重新拾起了,对python的热情. 贴个地址: https://github.com/d1y/lovepack/blob/master/kuaishou.py 前戏说明 因为我近乎癫狂的喜欢一个 ...
- 用python写的一个自动卸载python包的脚本
import osplist=os.popen("pip list") # 执行windows cmd命令,获取所有包package列表,并获取返回结果到plist#跳过第1,2行 ...
- 基于c编写的关于随机生成四则运算的小程序
基于http://www.cnblogs.com/HAOZHE/p/5276763.html改编写的关于随机生成四则运算的小程序 github源码和工程文件地址:https://github.com/ ...
随机推荐
- 原创:跳坑指南——微信小程序真机预览跟本地不同的问题
微信小程序中出现最多的一个问题,就是真机跟本地不同:我简单列举一些我发现的原因,给大家参考,大家也可以把自己发现的东西回复给我,给我参考:本地看不到数据,就先让本地能看到数据,再看本帖.... 1:本 ...
- 【JUC】JDK1.8源码分析之ConcurrentSkipListMap(二)
一.前言 最近在做项目的同时也在修复之前项目的一些Bug,所以忙得没有时间看源代码,今天都完成得差不多了,所以又开始源码分析之路,也着笔记录下ConcurrentSkipListMap的源码的分析过程 ...
- SqlServer基础之(触发器)
概念: 触发器(trigger)是SQL server 提供给程序员和数据分析员来保证数据完整性的一种方法,它是与表事件相关的特殊的存储过程,它的执行不是由程序调用,也不是手工启动,而是由事件来触 ...
- “全能”选手—Django 1.10文档中文版Part3
Django 1.10官方文档的入门教程已经翻译完毕,后续的部分将不会按照顺序进行翻译,而是挑重点的先翻译. 有兴趣的可以关注我的博客. 第一部分传送门 第二部分传送门 第四部分传送门 3.2 模型和 ...
- sealed、new、virtual、abstract与override 趣解
1. sealed——“断子绝孙” 密封类不能被继承.密封方法可以重写基类中的方法,但其本身不能在任何派生类中进一步重写.当应用于 方法或属性时,sealed修饰符必须始终与override一起使用. ...
- SQL 性能优化-查询优化(like查询)
废话不说,上代码 SET STATISTICS IO ON SELECT * FROM dbo.T_AssNews WHERE Content LIKE '%会%' 花费时间 执行计划 一个百分号的代 ...
- ASP.NET多种不同页面间数据传递的方法
1. Get(即使用QueryString显式传递) 方式:在url后面跟参数. 特点:简单.方便. 缺点:字符串长度最长为255个字符:数据泄漏在url中. 适用数据 ...
- WPF平台Grid控件性能比较
WPF官方发布第一个版本至今已经有10年了, 我们几乎在同时也开始了XAML开发.即使经过多年打造,我们依旧尝试提高:我们真的成功打造了高效灵活的控件吗?我没有在其他地方找到任何关于优秀的WPF表格性 ...
- Java时间和时间戳的相互转换
时间转换为时间戳: /* * 将时间转换为时间戳 */ public static String dateToStamp(String s) throws ParseException{ String ...
- IDE有毒
程序员按项目性质大致有三种:写Demo的.写Proto的.写成品的:按项目开发周期大致有:写开头的.写中间的.写结尾的. Demo是样品,主要是表面上初步实现,临时忽悠客户用的,不一定要求继续演化: ...