SQL Server 2008中如何为XML字段建立索引
from:http://blog.csdn.net/tjvictor/article/details/4370771
SQL Server中的XML索引分为两类:主XML 索引和辅助XML索引。其中辅助XML索引又分为:PATH 辅助XML索引,VALUE 辅助XML索引,PROPERTY辅助XML索引。
创建XML索引的语法示例如下:
- create table XMLTable(Id int primary key, XMLCol xml);
- go
- --XML主索引
- create primary xml index IPXML_XMLTable_XMLCol on XMLTable(XMLCol);
- --XML路径辅助索引
- create xml index IXML_XMLTable_XMLCol_Path on XMLTable(XMLCol)
- using xml index IPXML_XMLTable_XMLCol for path
- --XML属性辅助索引
- create xml index IXML_XMLTable_XMLCol_Property on XMLTable(XMLCol)
- using xml index IPXML_XMLTable_XMLCol for Property
- --XML内容辅助索引
- create xml index IXML_XMLTable_XMLCol_value on XMLTable(XMLCol)
- using xml index IPXML_XMLTable_XMLCol for value
需要注意是的,建立XML索引的表必须有主键。
建立索引的好处是提高查询效率,坏处是增加存储空间。下面结合实例,说明一下:
1.首先建立测试表,在SSMS中执行如下SQL语句create table XMLTable(Id int primary key, XMLCol xml);建表。
2.下面的程序是给表添加60万条数据,方便测试性能。至于为什么用程序添加而不是用insert语句,请参见我的另一篇博客:SQL Server 批量插入数据的两种方法http://blog.csdn.net/tjvictor/archive/2009/07/18/4360030.aspx
- static void Main(string[] args)
- {
- DataTable dt = GetTableSchema();
- for (int count = 1; count <= 600000; count++)
- {
- DataRow r = dt.NewRow();
- r[0] = count;
- r[1] = GetPropertyXML();
- dt.Rows.Add(r);
- }
- BulkToDB(dt);
- Console.WriteLine("finished");
- Console.ReadLine();
- }
- public static void BulkToDB(DataTable dt)
- {
- SqlConnection sqlConn = new SqlConnection(
- ConfigurationManager.ConnectionStrings["ConnStr1"].ConnectionString);
- SqlBulkCopy sqlBulkCopy = new SqlBulkCopy(sqlConn);
- sqlBulkCopy.BulkCopyTimeout = 0;
- sqlBulkCopy.BatchSize = dt.Rows.Count;
- sqlBulkCopy.DestinationTableName = "XMLTable";
- try
- {
- sqlConn.Open();
- if (dt != null && dt.Rows.Count != 0)
- {
- sqlBulkCopy.WriteToServer(dt);
- }
- }
- catch (Exception ex)
- {
- throw ex;
- }
- finally
- {
- sqlConn.Close();
- }
- }
- public static DataTable GetTableSchema()
- {
- DataTable dt = new DataTable();
- dt.Columns.AddRange(new DataColumn[]{
- new DataColumn("Id",typeof(int)),
- new DataColumn("XMLCol",typeof(string))});
- return dt;
- }
- public static int GetRandRange(int start, int end)
- {
- Random random = new Random(Guid.NewGuid().GetHashCode());
- return random.Next(start, end);
- }
- public static string GetPropertyXML()
- {
- StringBuilder buffer = new StringBuilder();
- buffer.AppendLine("<TJVICTOR>");
- for (int count = 0; count < GetRandRange(1, 10); count++)
- {
- int baseNum = GetRandRange(1, 100);
- buffer.AppendLine(string.Format("<Item{0} v=/"Property{0}/">Value{0}</Item{0}>", baseNum));
- }
- buffer.AppendLine("</TJVICTOR>");
- return buffer.ToString();
- }
3.执行一条查询语句,注意它的执行时间和执行计划:
select Id from XMLTable
where XMLCol.exist('/TJVICTOR/Item3')=1
由于机器配置不同,所以执行时间不会完全一样,这里只给出执行计划,以供参考:

所有时间都花在了Table Valued Function上,而且还是clustered index scan。
4.给这个表的XML字段加上索引。
- --XML主索引
- create primary xml index IPXML_XMLTable_XMLCol on XMLTable(XMLCol);
- --XML路径辅助索引
- create xml index IXML_XMLTable_XMLCol_Path on XMLTable(XMLCol)
- using xml index IPXML_XMLTable_XMLCol for path
- --XML属性辅助索引
- create xml index IXML_XMLTable_XMLCol_Property on XMLTable(XMLCol)
- using xml index IPXML_XMLTable_XMLCol for Property
- --XML内容辅助索引
- create xml index IXML_XMLTable_XMLCol_value on XMLTable(XMLCol)
- using xml index IPXML_XMLTable_XMLCol for value
注意:由于我们表中已经有60万条数据,所以建索引时间会很久,而且会占用大量内存和磁盘,本人就花费了10分钟左右,占了1G内存,和1.3G磁盘。请大家建索引时注意自己的硬盘空间,或者修改前面插入数据的程序,少插入一些数据。
5.重新执行上面的Sql语句:
select Id from XMLTable
where XMLCol.exist('/TJVICTOR/Item3')=1
你会发现,瞬间就出结果了,下面是执行计划,用到了XML index seek。

总结:建立XML索引后,查询效率会大大提高,经过本人的测试,xml.exist的执行效率最高,基本上提高了一个数据级,其它语句比如xml.query,xml.value等,查询速度提高了一倍左右,但总体不是太理想。但同时也发现,xml索引太占空间,比如上面的60万条记录吧,空间占用比例如下:
name rows reserved data index_size unused
XMLTable 600000 1479688 KB 160952 KB 1318184 KB 552 KB
关于xml.exist,xml.query,xml.value等基本使用方法,请参考下面的文章:
http://blog.csdn.net/tjvictor/archive/2009/07/21/4368511.aspx
如需转载,请注明本文原创自CSDN TJVictor专栏:
http://blog.csdn.net/tjvictor/archive/2009/07/22/4370771.aspx
SQL Server 2008中如何为XML字段建立索引的更多相关文章
- SQL Server 2008中新增的 1.变更数据捕获(CDC) 和 2.更改跟踪
概述 1.变更数据捕获(CDC) 每一次的数据操作都会记录下来 2.更改跟踪 只会记录最新一条记录 以上两种的区别: http://blog.csdn.n ...
- SQL Server 2008中新增的变更数据捕获(CDC)和更改跟踪
来源:http://www.cnblogs.com/downmoon/archive/2012/04/10/2439462.html 本文主要介绍SQL Server中记录数据变更的四个方法:触发器 ...
- 利用Ring Buffer在SQL Server 2008中进行连接故障排除
原文:利用Ring Buffer在SQL Server 2008中进行连接故障排除 出自:http://blogs.msdn.com/b/apgcdsd/archive/2011/11/21/ring ...
- SQL Server 2008中的CDC(Change Data Capture)功能使用及释疑
SQL Server 2008中的CDC(Change Data Capture)功能使用及释疑 关键词:CDC 原文:http://www.cnblogs.com/chenxizhang/arc ...
- SQL Server 2008中的MERGE(不仅仅是合并)
SQL Server 2008中的MERGE语句能做很多事情,它的功能是根据源表对目标表执行插入.更新或删除操作.最典型的应用就是进行两个表的同步. 下面通过一个简单示例来演示MERGE语句的使用方法 ...
- SQL Server 2008中的MERGE(数据同步)
OK,就像标题呈现的一样,SQL Server 2008中的MERGE语句能做很多事情,它的功能是根据源表对目标表执行插入.更新或删除操作.最典型的应用就是进行两个表的同步. 下面通过一个简单示例来演 ...
- SQL SERVER 2008中使用VARBINARY(MAX)进行图像存取的实现方法
在数据库应用项目开发中,经常会使用一些二进制的图像数据,存储和读取显示图像数据主要采用的是路径链接法和内存流法.路径链接法是将图像文件保存在固定的路径下,数据库中只存储图像文件的路径和名称 ...
- SQL Server 2008中的数据压缩
SQL Server 2008中引入了数据压缩的功能,允许在表.索引和分区中执行数据压缩.这样不仅可以大大节省磁盘的占用空间,还允许将更多数据页装入内存中,从而降低磁 盘IO,提升查询的性能.当然,凡 ...
- SQL Server 2008中增强的"汇总"技巧
本文转载:http://www.cnblogs.com/downmoon/archive/2012/04/06/2433988.html SQL Server 2008中的Pivot和UnPivot: ...
随机推荐
- Yii2 关于时间格式的用法
先添加配置文件: 'language' => 'zh-CN', 'timeZone' => 'Asia/Shanghai', 'components' => [ 'formatter ...
- win7安装centos7双系统
采用硬盘安装 前景 打算用U盘安装,但是u盘是FAT32格式限制了文件4g大小,我官网下的iso镜像大于4g,只好采用硬盘安装. 其实U盘安装是最方便的,网上很多教程用UltraISO软件把U盘直接作 ...
- 在线安装eclipse中html/jsp/xml editor插件 eclipseeditor
1.打开eclipse中的help————>Install New Software 2.点击Add按钮,然后弹出一个框,第一个文本框可以随便写,第二个一定要写: http://download ...
- EJB学习笔记之十(BMT事务和CMT事务)
1.前言 前两篇博客主要介绍了与事务相关的知识.比如事务的一些特性,以及并发产生的问题.本篇来解说一下EJB中两种处理事务的方式.一种是以生命式方式来管理事务(CMT):还有一种则是在EJB内部使用 ...
- Js中数组的追加
Concat arrayObject.concat(arrayX,arrayX,......,arrayX) 常用于 加载更多 ,数组的追加.
- c++解释--百度百科
c++ C++是在C语言的基础上开发的一种面向对象编程语言,应用广泛:C++支持多种编程范式 --面向对象编程.泛型编程和过程化编程.最新正式标准C++于2014年8月18日公布.[1] 其编程领域 ...
- NYOJ1097 Ugly Numbers 【丑数】
Ugly Numbers 时间限制:1000 ms | 内存限制:65535 KB 难度:2 描写叙述 Ugly numbers are numbers whose only prime fact ...
- CentOS 7上安装Zabbix(高速安装监控工具Zabbix)
前提要求(optional) 安装Zabbix监控工具前,先安装必要的执行工具包 yum install gcc gcc-c++ make openssl-devel curl wget net-sn ...
- JSON.parse 的用法,在js中用的。也是反序列化用法。
参数 text 必需. 一个有效的 JSON 字符串. reviver 可选. 一个转换结果的函数. 将为对象的每个成员调用此函数. 如果成员包含嵌套对象,则先于父对象转换嵌套对象. 对于每个成员,会 ...
- iOSeasy造成循引用的场景
场景一 :NStimer timer就是一个能在从如今開始的未来的某一个时刻又或者周期性的运行我们指定的方法的对象 NSTimer运行的必要条件:相应线程的RunLoop要开启,mode要相应 以下看 ...