Sqlserver 2005使用XML一次更新多条记录的方法
网络编程 2021-07-05 16:33www.168986.cn编程入门
Sqlserver 2005使用XML一次更新多条记录的方法,需要一次更新多条记录的朋友可以参考下。
我想很多人都知道,在oracle里面,存储过程里面可以传入数组(如int[]),也就是说,可以传多条记录到数据,从而一起更新。减少数据库的请求次数。
但SqlServer呢?bulk Insert这个很多人都知道,我也知道,但可惜,我从来没用过,只有导数据的时候才会考虑,但导数据DTS不是更方便吗?
手头的一个项目,有几个功能,每次需要更新N(N<1000)条记录,记录不多,但如果每次只更新一条,循环insert,那每个功能需要N次请求数据库,如果有1000个并发,那数据库除了做你这个事情,其他的活不用干了。所以,需要尽量减少数据库请求,做到一次更新所有的记录。
幸好,SqlServer给我们提供了一个新功能,利用XML(2000好像是没有这个功能的)。
先来假定一个这样的需求用户更新一个book,需要更新N个章节。
一般的思路是这样,先更新book,然后循环章节数,N次更新数据的章节表。大家可以看下这个性能。
那我们用XML试试
利用XML更新的存储过程
Create PROCEDURE UP_Book_Insert
(
@BookId INT,
@ChapterXml XML
)
AS
BEGIN
CREATE TABLE #table
(
ChapterId INT,
ChapterName VARCHAR(255),
Price INT
);
INSERT #table
SELECT
FROM (
SELECT X.C.value('Id[1]', 'int') AS ChapterId,
X.C.value('Name[1]', 'varchar(255)') AS ChapterName,
X.C.value('Price[1]','int') AS Price
FROM @ChapterXml.nodes('Chapter') AS X(C) --注意这里的X(C)命名空间是需要的
) t;
INSERT INTO tbChapter(BookId,ChapterId,ChapterName,Price)
SELECT @BookId,ChapterId,ChapterName,Price from #table;
END
其实,在存储过程里面可以把临时表去掉的。
然后我们执行下看看
执行存储过程
exec UP_Book_Insert 10000,'<Chapter><Id>268</Id><Name>第268章</Name><Price>100</Price></Chapter><Chapter><Id>273</Id><Name>第273章</Name><Price>100</Price></Chapter><Chapter><Id>275</Id><Name>第275章</Name><Price>100</Price></Chapter>'
怎么样?不错吧。只需要在存储过程里面对XML格式进行解析。
而在c#里面,XML格式可以传入DbType.String类型就可以了。
再写一个函数来生成XML格式的字符串
生成XML格式的函数
public static string FormatXmlInfo(List<ChapterInfo> list)
{
if (list==null||list.Count<=0)
{
return String.Empty;
}
StringBuilder sb = new StringBuilder();
foreach (ChapterInfo info in list)
{
sb.AppendFormat("<Chapter><Id>{0}</Id><Name>{1}</Name><Price>{2}</Price></Chapter>", info.ChapterId, info.ChapterName, info.Price);
}
return sb.ToString();
}
好了,完成了。
性能具体怎么样,还没进行测试,但肯定的一点是,比多次请求数据库,或者在存储过程里面循环分割字符串效率要高。
但SqlServer呢?bulk Insert这个很多人都知道,我也知道,但可惜,我从来没用过,只有导数据的时候才会考虑,但导数据DTS不是更方便吗?
手头的一个项目,有几个功能,每次需要更新N(N<1000)条记录,记录不多,但如果每次只更新一条,循环insert,那每个功能需要N次请求数据库,如果有1000个并发,那数据库除了做你这个事情,其他的活不用干了。所以,需要尽量减少数据库请求,做到一次更新所有的记录。
幸好,SqlServer给我们提供了一个新功能,利用XML(2000好像是没有这个功能的)。
先来假定一个这样的需求用户更新一个book,需要更新N个章节。
一般的思路是这样,先更新book,然后循环章节数,N次更新数据的章节表。大家可以看下这个性能。
那我们用XML试试
利用XML更新的存储过程
代码如下:
Create PROCEDURE UP_Book_Insert
(
@BookId INT,
@ChapterXml XML
)
AS
BEGIN
CREATE TABLE #table
(
ChapterId INT,
ChapterName VARCHAR(255),
Price INT
);
INSERT #table
SELECT
FROM (
SELECT X.C.value('Id[1]', 'int') AS ChapterId,
X.C.value('Name[1]', 'varchar(255)') AS ChapterName,
X.C.value('Price[1]','int') AS Price
FROM @ChapterXml.nodes('Chapter') AS X(C) --注意这里的X(C)命名空间是需要的
) t;
INSERT INTO tbChapter(BookId,ChapterId,ChapterName,Price)
SELECT @BookId,ChapterId,ChapterName,Price from #table;
END
其实,在存储过程里面可以把临时表去掉的。
然后我们执行下看看
执行存储过程
代码如下:
exec UP_Book_Insert 10000,'<Chapter><Id>268</Id><Name>第268章</Name><Price>100</Price></Chapter><Chapter><Id>273</Id><Name>第273章</Name><Price>100</Price></Chapter><Chapter><Id>275</Id><Name>第275章</Name><Price>100</Price></Chapter>'
怎么样?不错吧。只需要在存储过程里面对XML格式进行解析。
而在c#里面,XML格式可以传入DbType.String类型就可以了。
再写一个函数来生成XML格式的字符串
生成XML格式的函数
代码如下:
public static string FormatXmlInfo(List<ChapterInfo> list)
{
if (list==null||list.Count<=0)
{
return String.Empty;
}
StringBuilder sb = new StringBuilder();
foreach (ChapterInfo info in list)
{
sb.AppendFormat("<Chapter><Id>{0}</Id><Name>{1}</Name><Price>{2}</Price></Chapter>", info.ChapterId, info.ChapterName, info.Price);
}
return sb.ToString();
}
好了,完成了。
性能具体怎么样,还没进行测试,但肯定的一点是,比多次请求数据库,或者在存储过程里面循环分割字符串效率要高。
编程语言
- 如何快速学会编程 如何快速学会ug编程
- 免费学编程的app 推荐12个免费学编程的好网站
- 电脑怎么编程:电脑怎么编程网咯游戏菜单图标
- 如何写代码新手教学 如何写代码新手教学手机
- 基础编程入门教程视频 基础编程入门教程视频华
- 编程演示:编程演示浦丰投针过程
- 乐高编程加盟 乐高积木编程加盟
- 跟我学plc编程 plc编程自学入门视频教程
- ug编程成航林总 ug编程实战视频
- 孩子学编程的好处和坏处
- 初学者学编程该从哪里开始 新手学编程从哪里入
- 慢走丝编程 慢走丝编程难学吗
- 国内十强少儿编程机构 中国少儿编程机构十强有
- 成人计算机速成培训班 成人计算机速成培训班办
- 孩子学编程网上课程哪家好 儿童学编程比较好的
- 代码编程教学入门软件 代码编程教程