MS SQL Server:分区表、分区索引 详解
1. 分区表简介
使用分区表的主要目的,是为了改善大型表以及具有各种访问模式的表的可伸缩性和可管理性。
大型表:数据量巨大的表。
访问模式:因目的不同,需访问的不同的数据行集,每种目的的访问可以称之为一种访问模式。
分区一方面可以将数据分为更小、更易管理的部分,为提高性能起到一定的作用;另一方面,对于如果具有多个CPU的系统,分区可以是对表的操作通过并行的方式进行,这对于提升性能是非常有帮助的。
注意:只能在 SQL Server Enterprise Edition 中创建分区函数。只有 SQL Server Enterprise Edition 支持分区。
2. 创建分区表或分区索引的步骤
可以分为以下步骤:
1. 确定分区列和分区数
2. 确定是否使用多个文件组
3. 创建分区函数
4. 创建分区架构(Schema)
5. 创建分区表
6. 创建分区索引
下面详细描述的创建分区表、分区索引的步骤。
2.1. 确定分区列和分区数
在开始做分区操作之前,首先要确定待分区表的访问模式,该模式决定了什么列适合做分区键。例如,对于销售数据,一般会先根据日期把数据范围限定在一个范围内,然后在这个基础上做进一步的查询,这样,就可以把日期作为分区列。
确定了分区列之后,需要进一步确定分区数,亦即分区表中需要包含多少数据,每个分区的数据应该限定在哪个范围。
2.2. 确定是否使用多个文件组
为了有助于优化性能和维护,应该使用文件组分离数据。一般情况下,如果经常对分区的整个数据集操作,则文件组数最好与分区数相同,并且这些文件组通常应该位于不同的磁盘上,再配合多个CPU,则SQL Server 可以并行处理多个分区,从而大大缩短处理大量复杂报表和分析的总体时间。
2.3. 创建分区函数
分区函数用于定义分区的边界条件,创建分区函数的语法如下:
CREATEPARTITIONFUNCTIONpartition_function_name ( input_parameter_type )
ASRANGE[LEFT | RIGHT]
FORVALUES([boundary_value [ ,...n]] )
[;]
参数说明:
partition_function_name
是分区函数的名称。分区函数名称在数据库内必须唯一,并且符合标识符的规则。
input_parameter_type
是用于分区的列的数据类型。当用作分区列时,除text、ntext、image、xml、timestamp、varchar(max)、nvarchar(max)、varbinary(max)、别名数据类型或 CLR 用户定义数据类型外,所有数据类型均有效。
实际列(也称为分区列)是在CREATETABLE或CREATEINDEX语句中指定的。
boundary_value
为使用 partition_function_name 的已分区表或索引的每个分区指定边界值。如果 boundary_value 为空,则分区函数使用 partition_function_name 将整个表或索引映射到单个分区。只能使用CREATETABLE或CREATEINDEX语句中指定的一个分区列。
boundary_value 是可以引用变量的常量表达式。这包括用户定义类型变量,或函数以及用户定义函数。它不能引用 Transact-SQL 表达式。boundary_value 必须与 input_parameter_type 中提供的数据类型相匹配或者可隐式转换为该数据类型,并且如果该值的大小和小数位数与 input_parameter_type 中相应的值的大小和小数位数不匹配,则在隐式转换过程中该值不能被截断。
注意:
如果 boundary_value 包含datetime或smalldatetime文字值,则为这些文字值在计算时假设 us_english 是会话语言。不推荐使用此行为。要确保分区函数定义对于所有会话语言都具有预期的行为,建议使用对于所有语言设置都以相同方式进行解释的常量,例如 yyyymmdd 格式;或者将文字值显式转换为特定样式。有关详细信息,请参阅编写国际化 Transact-SQL 语句。若要确定服务器的语言会话,请运行SELECT@@LANGUAGE。
...n
指定 boundary_value 提供的值的数目,不能超过999。所创建的分区数等于 n+1。不必按顺序列出各值。如果值未按顺序列出,则 Microsoft SQL Server2005数据库引擎将对它们进行排序,创建函数并返回一个警告,说明未按顺序提供值。如果 n 包括任何重复的值,则数据库引擎将返回错误。
LEFT|RIGHT
指定当间隔值由 数据库引擎 按升序从左到右排序时,boundary_value[,...n]属于每个边界值间隔的哪一侧(左侧还是右侧)。如果未指定,则默认值为LEFT。
创建分区函数示例:
CREATEPARTITIONFUNCTIONPF_Left(int)
ASRANGELEFT
FORVALUES(10,20)
GO
CREATEPARTITIONFUNCTIONPF_Right(int)
ASRANGELEFT
FORVALUES(10,20)
GO
PF_Left 和 PF_Right 分区函数的区分:
分区函数 分区1 分区2 分区3
PF_Left<=10>10and<=20>20
PF_Right<10>=10and<20>=20
2.4. 创建分区架构(Schema)
创建分区函数后,必须将其与分区架构(Schema)相关联,以便将分区定向至特定的文件组。定义分区架构师,即使多个分区位于同一个文件组中,也必须为每个分区指定一个文件组。
创建分区架构的语法如下:
GOCREATE PARTITION SCHEME partition_scheme_name
ASPARTITION partition_function_name
[ALL]TO( { file_group_name|[PRIMARY]}[,...n])
[;]
参数:
partition_scheme_name
分区方案的名称。分区方案名称在数据库中必须是唯一的,并且符合标识符规则。
partition_function_name
使用分区方案的分区函数的名称。分区函数所创建的分区将映射到在分区方案中指定的文件组。partition_function_name 必须已经存在于数据库中。
ALL
指定所有分区都映射到在 file_group_name 中提供的文件组,或映射到主文件组(如果指定了[PRIMARY]。如果指定了ALL,则只能指定一个 file_group_name。
file_group_name|[PRIMARY][,...n]
指定用来持有由 partition_function_name 指定的分区的文件组的名称。file_group_name 必须已经存在于数据库中。
如果指定了[PRIMARY],则分区将存储于主文件组中。如果指定了ALL,则只能指定一个 file_group_name。分区分配到文件组的顺序是从分区1开始,按文件组在[,...n]中列出的顺序进行分配。在[,...n]中,可以多次指定同一个 file_group_name。如果 n 不足以拥有在 partition_function_name 中指定的分区数,则CREATEPARTITION SCHEME 将失败,并返回错误。
如果 partition_function_name 生成的分区数少于文件组数,则第一个未分配的文件组将标记为NEXTUSED,并且出现显示命名NEXTUSED 文件组的信息。如果指定了ALL,则单独的 file_group_name 将为该 partition_function_name 保持它的NEXTUSED 属性。如果在ALTERPARTITIONFUNCTION语句中创建了一个分区,则NEXTUSED 文件组将再接收一个分区。若要再创建一个未分配的文件组来拥有新的分区,请使用ALTERPARTITION SCHEME。
在 file_group_name[1,...n]中指定主文件组时,必须像在[PRIMARY]中那样分隔PRIMARY,因为它是关键字。
创建分区架构示例:
CREATEPARTITIONFUNCTIONmyRangePF1 (int)
ASRANGELEFTFORVALUES(1,100,1000);
GO
CREATEPARTITION SCHEME myRangePS1
ASPARTITION myRangePF1
TO(test1fg, test2fg, test3fg, test4fg);
GO
2.5. 创建分区表
定义了分区函数(逻辑结构)和分区架构(物理结构)后,既可以创建分区表来利用它们。分区表定义应使用的分区架构,而分区架构又定义其使用的分区函数。要将这三者结合起来,必须指定应用于分区函数的列 。范围分区始终只映射到表中的一列。
CREATETABLE语法如下:
CREATETABLE
[database_name . [ schema_name].|schema_name . ] table_name
( {<column_definition>|<computed_column_definition>}
[<table_constraint>][,...n])
[ON { partition_scheme_name ( partition_column_name ) | filegroup
| "default" }]
[{ TEXTIMAGE_ON { filegroup | "default" }]
[;]
示例如下:
CREATETABLEmyRangePT1
(
IDintnotnull,
AGEint,
PRIMARYKEY(ID)
)ONmyRangePS1(myRangePF1)
GO
2.6. 创建分区索引
索引对于提高查询性能非常有效,因此,一般应该考虑应该考虑为分区表建立索引,为分区表建立索引与为普通表建立索引的语法一直,但是,其行为与普通索引有所差异。
默认情况下,分区表中创建的索引使用与分区表相同分区架构和分区列,这样,索引将于表对齐。将表与其索引对齐,可以使管理工作更容易进行,对于滑动窗口方案尤其如此。若要启动分区切换,表的所有索引都必须对齐。
在创建索引时,也可以指定不同的分区方案(Schema)或单独的文件组(FileGroup)来存储索引,这样SQL Server 不会将索引与表对齐。
在已分区的表上创建索引(分区索引)时,应该注意以下事项:
唯一索引
建立唯一索引(聚集或者非聚集)时,分区列必须出现在索引列中。此限制将使SQL Server只调查单个分区,并确保表中宠物的新键值。如果分区依据列不可能包含在唯一键中,则必须使用DML触发器,而不是强制实现唯一性。
非唯一索引
对非唯一的聚集索引进行分区时,如果未在聚集键中明确指定分区依据列,默认情况下SQL Server 将在聚集索引列中添加分区依据列。
对非唯一的非聚集索引进行分区时,默认情况下SQL Server 将分区依据列添加为索引的包含性列,以确保索引与基表对齐,若果索引中已经存在分区依据列,SQL Server 将不会像索引中添加分区依据列。
3. 分区操作
分区适用于可以缩放的大型表,所以随着时间和环境的变化,就会产生对分区的拆分、合并、移动的需求。
3.1. 拆分与合并分区
通过拆分或合并边界值更改分区函数。通过执行ALTERPARTITIONFUNCTION,可以将使用分区函数的任何表或索引的某个分区拆分为两个分区,也可以将两个分区合并为一个分区。
注意:多个表或索引可以使用同一分区函数。ALTERPARTITIONFUNCTION在单个事务中影响所有这些表或索引。
ALTERPARTITIONFUNCTION语法如下:
ALTERPARTITIONFUNCTIONpartition_function_name()
{
SPLIT RANGE ( boundary_value )
|MERGE RANGE ( boundary_value )
}[;]
参数说明:
partition_function_name
要修改的分区函数的名称。
SPLIT RANGE ( boundary_value )
在分区函数中添加一个分区。boundary_value 确定新分区的范围,因此它必须不同于分区函数的现有边界范围。根据 boundary_value,Microsoft SQL Server2005数据库引擎将某个现有范围拆分为两个范围。在这两个范围中,新 boundary_value 所在的范围被视为是新分区。
重要提示:
文件组必须处于联机状态,并且必须由使用此分区函数的分区方案标记为NEXTUSED,以保存新分区。在CREATEPARTITION SCHEME 语句中,将把文件组分配给分区。如果CREATEPARTITION SCHEME 语句分配了多余的文件组(在CREATEPARTITIONFUNCTION语句中创建的分区数少于用于保存它们的文件组),则存在未分配的文件组,分区方案将把其中的某个文件组标记为NEXTUSED。该文件组将保存新的分区。如果分区方案未将任何文件组标记为NEXTUSED,则必须使用ALTERPARTITION SCHEME 添加一个文件组或指定一个现有文件组来保存新分区。可以指定已保存分区的文件组来保存附加分区。由于一个分区函数可以参与多个分区方案,因此所有使用分区函数(您向其中添加了分区)的分区方案都必须拥有一个NEXTUSED 文件组。否则,ALTERPARTITIONFUNCTION将失败并出现错误,该错误显示缺少NEXTUSED 文件组的一个或多个分区方案。
MERGE[RANGE ( boundary_value)]
删除一个分区并将该分区中存在的所有值都合并到剩余的某个分区中。RANGE (boundary_value) 必须是一个现有边界值,已删除分区中的值将合并到该值中。如果最初保存 boundary_value 的文件组没有被剩余分区使用,也没有使用NEXTUSED 属性进行标记,则将从分区方案中删除该文件组。合并的分区驻留在最初不保存 boundary_value 的文件组中。boundary_value 是一个可以引用变量(包括用户定义类型变量)或函数(包括用户定义函数)的常量表达式。它无法引用 Transact-SQL 表达式。boundary_value 必须匹配或可以隐式转换为其对应列的数据类型,并且当值的大小和小数位数不匹配其对应 input_parameter_type 时,将无法在隐式转换过程中被截断。
示例:
ALTERPARTITION SCHEME PS_HistoryArchive
NEXTUSED[PRIMARY]
备注:
ALTERPARTITIONFUNCTION在单个原子操作中对使用该函数的任何表和索引进行重新分区。但该操作在脱机状态下进行,并且根据重新分区的范围,可能会消耗大量资源。
ALTERPARTITIONFUNCTION只能用于将一个分区拆分为两个分区,或将两个分区合并为一个分区。若要更改其他情况下对表进行分区方法(例如,将10个分区合并为5个分区),可以尝试使用以下任何选项。根据系统配置,这些选项可能在资源消耗方面有所不同:
使用所需的分区函数创建一个新的已分区表,然后使用INSERTINTO...SELECTFROM语句将旧表中的数据插入新表。
为堆创建分区聚集索引。
注意:
删除已分区的聚集索引将产生分区堆。
通过将 Transact-SQLCREATEINDEX语句与DROPEXISTING=ON子句一起使用来删除并重新生成现有的已分区索引。
执行一系列ALTERPARTITIONFUNCTION语句。
ALTERPARITITIONFUNCTION所影响的全部文件组都必须处于联机状态。
如果使用分区函数的任何表中存在已禁用的聚集索引,ALTERPARTITIONFUNCTION都将失败。
Microsoft SQL Server2005不对修改分区函数提供复制支持。必须在订阅数据库中手动应用对发布数据库中的分区函数的更改。
3.2. 移动分区数据
可以使用ALTERTABLE....... SWITCH 语句按一下方式快速有效地移动数据子集:
将某个表中的数据移动到另一个表中;
将某个表作为分区添加到现存的已分区表中;
将分区从一个已分区表切换到另一个已分区表;
删除分区以形成单个表。
使用这些方案移动数据时,无论集合有多大,此方案都能快速有效地进行传输,因为操作并不以物理方式移动数据,只有关于存储位置的元数据会从一个分区变为另一个分区。
ALTERTABLE.... SWITCH 的语法如下:
ALTERTABLE[database_name . [ schema_name].|schema_name . ] table_name
{
SWITCH[PARTITION source_partition_number_expression_r]
TO[schema_name].target_table
[PARTITION target_partition_number_expression_r]
}
[;]
----------------------------------------------
--创建示例
USE[master]
GO
ifexists(select*fromsys.databaseswherename='Test_1')
dropdatabaseTest_1
GO
--创建新库,要演练分区所以我们会多创建两个文件组Test_A,Test_B,以便在后面的分区方案中使用。
CREATEDATABASE[Test_1]ONPRIMARY
( NAME=N'test_1', FILENAME=N'D:\sqldata\test_1.mdf', SIZE=10240KB , MAXSIZE=UNLIMITED, FILEGROWTH=1024KB ),
FILEGROUP[test_A]
( NAME=N'Test_A', FILENAME=N'D:\sqldata\test_A.ndf', SIZE=1024KB , MAXSIZE=UNLIMITED, FILEGROWTH=1024KB ),
FILEGROUP[test_B]
( NAME=N'Test_B', FILENAME=N'D:\sqldata\test_B.ndf', SIZE=1024KB , MAXSIZE=UNLIMITED, FILEGROWTH=1024KB )
LOGON
( NAME=N'Test_log', FILENAME=N'D:\sqldata\Test_log.ldf', SIZE=7616KB , MAXSIZE=2048GB , FILEGROWTH=10%)
COLLATE Chinese_PRC_CI_AS
GO
USE[Test_1]
GO
--若分区函数存在则先drop掉
IFEXISTS(SELECT*FROMsys.partition_functionsWHEREname=N'test_partition')
DROPPARTITIONFUNCTION[test_partition]
GO
/**//*创建分区函数给后面的分区方案使用,分区函数很简单就是指定一个范围确定在某个值为什么的时候放在那个分区上*/
--新建一个简单的分区函数,该函数以1000为界分两个区
createpartitionfunctiontest_partition(int)
AS
RANGELEFTFORVALUES(1000)
go
/**//*看分区方案是否存在,若存在先drop掉*/
IFEXISTS(SELECT*FROMsys.partition_schemesWHEREname=N'test_scheme')
DROPPARTITION SCHEME test_scheme
GO
--创建分区方案,分区方案需要指定一个分区函数,并指定在分区函数中分的区需要放在哪一个文件组上
createpartition scheme test_scheme
AS
PARTITION[test_partition]TO(test_A,test_B)
GO
--创建分区表
ifobject_id('student','U')isnotnull
droptablestudent;
go
createtablestudent
(
idintidentity(1,1)notnull,
namevarchar(10)notnull,
classintnotnull,
gradeint
)ontest_scheme(class)--在此处指定该表要使用的分区方案,并将指定分区依据列
go
--随便插入几条数据
insertintostudentvalues('AQU',10,100);--这条数据在A分区上
insertintostudentvalues('AQU_边界',1000,89);--这边数据也在A分区上是个边界,因为我们上面在函数中指定的是RANGE LEFT,所以1000在A分区上
insertintostudentvalues('BQU',1001,90);--这一条肯定是在B分区上了。
go
--最后看看结果。$partition.分区函数(分区列)可以返回某一行所在的分区序号
select*,分区序号=$partition.test_partition(class)fromstudent
GO