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: ...
随机推荐
- Yii添加验证码
添加带验证码的登陆: 1.先在模型modules下的LoginForm.php定义一个存储验证码的变量:public $verfyCode: 2.然后在rules()方法里定义:array('veri ...
- 【Swift】学习笔记(三)——字符和字符串
基本了解了变量的创建和基础数据类型,但是在开发中用得最多的还是字符串.那什么是字符串呢? Swift 的String类型表示特定序列的Character(字符) 类型值的集合,它是值类型具有可变性 S ...
- Mysql的Root密码忘记,查看或修改的解决方法(图文介绍)
http://www.jb51.net/article/38473.htm 首先启动命令行 1.在命令行运行:taskkill /f /im mysqld-nt.exe 下面的操作是操作mysql中b ...
- DOM操作,控制HTML元素 (原生JS)
文档对象模型DOM(Document Object Model)定义访问和处理HTML文档的标准方法.DOM 将HTML文档呈现为带有元素.属性和文本的树结构(节点树). 先来看看下面代码: 将HTM ...
- 【Java】Java_06 基本数据类型
1.基本数据类型 Java是一种强类型语言,每个变量都必须声明其类型. . Java的数据类型分为两大类:基本类型(primitive type)和引用类型 (reference type) ...
- WebLogic92数据源配置
一. 将数据库连接所需的包导入(非常重要) 最简单的方法就是,将所需jar包复制至%MYDOMAIN_HOME%/lib中,约定本应用域的名称为“ MyDomain”,根路径为%MYDOMAIN_HO ...
- Velocity中文乱码问题解决方法
http://yonge812.iteye.com/blog/1010290 Velocity中文乱码问题需要注意一下几点: 1.eclipse默认编码方式 2.页面的编码方式 3.volocity模 ...
- js获取100个随机数存入数组
. //js获取100个随机数存入数组 $(function () { var arr = []; ; var str = ""; ) { , ); ) { arr[num] = ...
- 【CODEFORCES】 C. Captain Marmot
C. Captain Marmot time limit per test 1 second memory limit per test 256 megabytes input standard in ...
- CentOS-6.*安装配置SVN
安装说明 系统环境:CentOS-6.3 安装方式:yum install (源码安装容易产生版本兼容的问题) 安装软件:系统自动下载SVN软件 检查已安装版本 #检查是否安装了低版本的SVN [ro ...