SQL Server 2005及以上版本中XML操作函数详解打赏

一直以来,使用sqlserver都比较少,之前读书时学过sqlserver2000、sqlserver2005,却一直没接触到类似于sqlserver2005中引入新的字段类型XML、varchar(max)、nvarchar(max)、varbinary(max)之类的知识。直到今天,用到大字节的数据存储,才发现有这个改进,确切点说,是sqlserver2005中废弃了text、ntext 和image类型采用新的max标记来扩展原有字段的存储能力,如varchar(max)、nvarchar(max)、varbinary(max),关于这些在后面单独描述,本篇介绍一下sqlserver2005以后新增的一个类型——XML。

我们采用T-Sql操作Xml数据。使用sqlserver2005引入的XML数据类型。用户可以创建这样的表,它在关系列之外还有一个或多个 XML 类型的列;此外,还允许带有变量和参数。为了更好地支持 XML 模型特征(例如文档顺序和递归结构),XML 值以内部格式存储为大型二进制对象 (BLOB)。

用户将一个XML数据存入数据库的时候,可以使用这个XML的字符串,SQL Server会自动的将这个字符串转化为XML类型,并存储到数据库中。随着SQL Server 对XML字段的支持,相应的,T-SQL语句也提供了大量对XML操作的功能来配合SQL Server中XML字段的使用。本文主要说明如何使用SQL语句对XML进行操作。

在进行数据库的设计中,我们可以在表设计器中,很方便的将一个字段定义为XML类型。需要注意的是,XML字段不能用来作为主键或者索引键。同样,我们也可以使用SQL语句来创建使用XML字段的数据表,下面的语句创建一个名为“docs”的表,该表带有整型主键“pk”和非类型化的 XML 列“xCol”:

CREATE TABLE docs (pk INT PRIMARY KEY, xCol XML not null)

XML类型除了在表中使用,还可以在存储过程、事务、函数等中出现。下面我们来完成我们对XML操作的第一步,使用SQL语句定义一个XML类型的数据,并为它赋值:

declare @xmlDoc xml;
set @xmlDoc='<book id="0001">
<title>C Program</title>
<author>David</author>
<price>21</price>
</book>'

在定义了一个XML类型的数据之后,我们最常用的就是查询操作,下面我们来介绍如何使用SQL语句来进行查询操作的。首先要明确一个基本原则,XML类型的数据之间以及XML类型与其它数据类型之间都是不能比较的,也就是说XML类型的数据不能出现在等号的任何一边。大致可分为查询类,修改类和跨域查询类。

查询类包含:query(),value(),exist()和nodes().
修改类包含:modify().
跨域查询类包含:sql:variable()和sql:column().

查询类

query()方法:返回满足条件的所有XML行。只能用于SELECT子句当中。
value()方法:返回从XML节点中提取的标量值。必须在value()方法的第二个参数中指定所返回的标量的数据类型,所以value()方法可以与其它标量进行比较。可用于SELECT子句和WHERE子句。
exist()方法:返回int型标量的0或者1。对每行的XML数据类型进行存在性检查。可用于SELECT子句和WHERE子句。
nodes()方法:返回只有一个栏位的table,且该table的栏位是XML数据类型。所以nodes()方法只能出现在FROM子句中。

在T-Sql中,提供了两个对XML类型数据进行查询的函数,分别是query(xquery)和value(xquery, dataType),其中,query(xquery)得到的是带有标签的数据,而value(xquery, dataType)得到的则是标签的内容。接下类我们分别使用这两个函数来进行查询。

1、使用query(xquery) 查询

我们需要得到书的标题(title),使用query(xquery)来进行查询,查询语句为:

select @xmlDoc.query('/book/title')

运行结果如图:
sqlserverxml1

2、使用value(xquery, dataType) 查询

同样是得到书的标题,使用value函数,需要指明两个参数,一个为xquery, 另一个为得到数据的类型。看下面的查询语句:

select @xmlDoc.value('(/book/title)[1]', 'nvarchar(max)')

运行结果如图:
sqlserverxml2

3、查询属性值

无论是使用query还是value,都可以很容易的得到一个节点的某个属性值,例如,我们很希望得到book节点的id,我们这里使用value方法进行查询,语句为:

select @xmlDoc.value('(/book/@id)[1]', 'nvarchar(max)')

运行结果如图:
sqlserverxml3

4、使用xpath进行查询

xpath是.net平台下支持的,统一的Xml查询语句。使用XPath可以方便的得到想要的节点,而不用使用where语句。例如,我们在@xmlDoc中添加了另外一个节点,重新定义如下:

set @xmlDoc='<root>
<book id="0001">
<title>C# Program</title>
<author>Jerry</author>
<price>50</price>
</book>
<book id="0002">
<title>Java Program</title>
<author>Tom</author>
<price>49</price>
</book>
</root>'

得到id为0002的book节点

select @xmlDoc.query('(/root/book[@id="0002"])')

上面的语句可以独立运行,它得到的是id为0002的节点。运行结果如下图:
sqlserverxml4

修改类

modify()方法:允许修改XML实例的某些部分,例如添加或删除子树,或者更新标量值(如将书的价格从9.99替换为39.99)。无返回值,只能用于SET子句中。

SQL的修改操作包括更新和删除。SQL提供了modify()方法,实现对Xml的修改操作。modify方法的参数为XML修改语言。XML修改语言类似于SQL 的Insert、Delete、UpDate,但并不一样。

1、修改节点值

我们希望将id为0001的书的价钱(price)修改为100, 我们就可以使用modify方法。代码如下:

set @xmlDoc.modify('replace value of (/root/book[@id=0001]/price/text())[1] with "100"')

得到id为0001的book节点

select @xmlDoc.query('(/root/book[@id="0001"])')

注意:modify方法必须出现在set的后面。运行结果如图:
sqlserverxml5

2、删除节点

接下来我们来删除id为0002的节点,代码如下:
删除节点id为0002的book节点

set @xmlDoc.modify('delete /root/book[@id=0002]')
select @xmlDoc

运行结果如图:
sqlserverxml6

3、添加节点

很多时候,我们还需要向xml里面添加节点,这个时候我们一样需要使用modify方法。下面我们就向id为0001的book节点中添加一个ISBN节点,代码如下:
添加节点

set @xmlDoc.modify('insert <isbn>78-596-134</isbn> before (/root/book[@id=0001]/price)[1]')
select @xmlDoc.query('(/root/book[@id="0001"]/isbn)')

运行结果如图:
sqlserverxml7

4、添加和删除属性

当你学会对节点的操作以后,你会发现,很多时候,我们需要对节点进行操作。这个时候我们依然使用modify方法,例如,向id为0001的book节点中添加一个date属性,用来存储出版时间。代码如下:
添加属性

set @xmlDoc.modify('insert attribute date{"2008-11-27"} into (/root/book[@id=0001])[1]')
select @xmlDoc.query('(/root/book[@id="0001"])')

运行结果如图:
sqlserverxml8
如果你想同时向一个节点添加多个属性,你可以使用一个属性的集合来实现,属性的集合可以写成:(attribute date{"2008-11-27"}, attribute year{"2008"}),你还可以添加更多。这里就不再举例了。

5、删除属性

删除一个属性,例如删除id为0001 的book节点的id属性,我们可以使用如下代码:
删除属性

set @xmlDoc.modify('delete root/book[@id="0001"]/@id')
select @xmlDoc.query('(/root/book)[1]')

运行结果如图:
sqlserverxml9

6、修改属性

修改属性值也是很常用的,例如把id为0001的book节点的id属性修改为0005,我们可以使用如下代码:
修改属性

set @xmlDoc.modify('replace value of (root/book[@id="0001"]/@id)[1] with "0005"')
select @xmlDoc.query('(/root/book)[1]')

运行结果如图:
sqlserverxml10

跨域查询类

sql:variable()方法:可以在XQuery或XML DML表达式中应用SQL变量的值。

如果数据驻留在关系数据类型的列和XML数据类型的列的组合中,就可能需要编写查询来组合关系数据处理和XML数据处理(组合的对象还可以是标量数据类型的SQL变量和XML数据),这样的查询就叫跨域查询。

sql:variable()方法:

DECLARE @isbn varchar(20)
SET     @isbn = '0-7356-1588-2'
SELECT  xCol
FROM    docs
WHERE   xCol.exist ('/book[@ISBN = sql:variable("@isbn")]') = 1

sql:column()方法:可以在XQuery或XML DML表达式中使用来自关系列的值。

整理来源:http://youring2.cnblogs.com/&http://developer.51cto.com/art/200902/111125.htm

SQL Server 2005及以上版本中XML操作函数详解
文章《SQL Server 2005及以上版本中XML操作函数详解》二维码
  • 微信打赏
  • 支付宝打赏

已有4条评论

  1. kitten0

    这个就会安装一下

    2012-12-24 16:04 回复

(必填)

(必填)

(可选)