`

数据库优化分区篇

 
阅读更多

 

大数据量的数据库设计准则:
  1、分区 (list、range、hash、key)
  2、索引

一、什么是表分区(以Mysql为例)
通俗地讲表分区是将一大表,根据条件分割成若干个小表mysql5.1开始支持数据表分区了。
如:某用户表的记录超过了600万条,那么就可以根据入日期将表分区,也可以根据所在地将表分区。当然也可根据其他的条件分区。 二、为什么要对表进行分区
为了改善大型表以及具有各种访问模式的表的可伸缩性,可管理性和提高数据库效率。分区的一些优点包括:
      1)、与单个磁盘或文件系统分区相比,可以存储更多的数据
      2)、对于那些已经失去保存意义的数据,通常可以通过删除与那些数据有关的分区,很容易地删除那些数据。相反地,在某些情况下,添加新数据的过程又可以通过为那些新数据专门增加一个新的分区,来很方便地实现。通常和分区有关的其他优点包括下面列出的这些。MySQL分区中的这些功能目前还没有实现,但是在我们的优先级列表中,具有高的优先级;我们希望在5.1的生产版本中,能包括这些功能。
      3)、一些查询可以得到极大的优化,这主要是借助于满足一个给定WHERE语句的数据可以只保存在一个或多个分区内,这样在查找时就不用查找其他剩余的分区。因为分区可以在创建了分区表后进行修改,所以在第一次配置分区方案时还不曾这么做时,可以重新组织数据,来提高那些常用查询的效率。
      4)、涉及到例如SUM()和COUNT()这样聚合函数的查询,可以很容易地进行并行处理。这种查询的一个简单例子如 “SELECT salesperson_id, COUNT (orders) as order_total FROM sales GROUP BY salesperson_id;”。通过“并行”,这意味着该查询可以在每个分区上同时进行,最终结果只需通过总计所有分区得到的结果。
      5)、通过跨多个磁盘来分散数据查询,来获得更大的查询吞吐量。
三、分区类型 · RANGE分区:基于属于一个给定连续区间的列值,把多行分配给分区。 
· LIST分区:类似于按RANGE分区,区别在于LIST分区是基于列值匹配一个离散值集合中的某个值来进行选择。 
· HASH分区:基于用户定义的表达式的返回值来进行选择的分区,该表达式使用将要插入到表中的这些行的列值进行计算。这个函数可以包含MySQL 中有效的、产生非负整数值的任何表达式。
· KEY分区:类似于按HASH分区,区别在于KEY分区只支持计算一列或多列,且MySQL 服务器提供其自身的哈希函数。必须有一列或多列包含整数值。
  • RANGE分区
       基于属于一个给定连续区间的列值,把多行分配给分区。       这些区间要连续且不能相互重叠,使用VALUES LESS THAN操作符来进行定义。以下是实例。 Sql代码
  1. word">CREATE TABLE employees (   
  2.     id INT NOT NULL,   
  3.     fname VARCHAR(30),   
  4.     lname VARCHAR(30),   
  5.     hired DATE NOT NULL DEFAULT '1970-01-01',   
  6.     separated DATE NOT NULL DEFAULT '9999-12-31',   
  7.     job_code INT NOT NULL,   
  8.     store_id INT NOT NULL  
  9. )   
  10.   
  11. partition BY RANGE (store_id) (   
  12.     partition p0 VALUES LESS THAN (6),   
  13.     partition p1 VALUES LESS THAN (11),   
  14.     partition p2 VALUES LESS THAN (16),   
  15.     partition p3 VALUES LESS THAN (21)   
  16. );   
       按照这种分区方案,在商店1到5工作的雇员相对应的所有行被保存在分区P0中,商店6到10的雇员保存在P1中,依次类推。注意,每个分区都是按顺序进行定义,从最低到最高。这是PARTITION BY RANGE 语法的要求;在这点上,它类似于C或Java中的“switch ... case”语句。
       对于包含数据(72, 'Michael', 'Widenius', '1998-06-25', NULL, 13)的一个新行,可以很容易地确定它将插入到p2分区中,但是如果增加了一个编号为第21的商店,将会发生什么呢?在这种方案下,由于没有规则把store_id大于20的商店包含在内,服务器将不知道把该行保存在何处,将会导致错误。 要避免这种错误,可以通过在CREATE TABLE语句中使用一个“catchall” VALUES LESS THAN子句,该子句提供给所有大于明确指定的最高值的值:
Sql代码
  1. CREATE TABLE employees (   
  2.     id INT NOT NULL,   
  3.     fname VARCHAR(30),   
  4.     lname VARCHAR(30),   
  5.     hired DATE NOT NULL DEFAULT '1970-01-01',   
  6.     separated DATE NOT NULL DEFAULT '9999-12-31',   
  7.     job_code INT NOT NULL,   
  8.     store_id INT NOT NULL  
  9. )   
  10.   
  11. PARTITION BY RANGE (store_id) (   
  12.     PARTITION p0 VALUES LESS THAN (6),   
  13.     PARTITION p1 VALUES LESS THAN (11),   
  14.     PARTITION p2 VALUES LESS THAN (16),   
  15.     PARTITION p3 VALUES LESS THAN MAXVALUE   
  16. );  
    MAXVALUE 表示最大的可能的整数值。现在,store_id 列值大于或等于16(定义了的最高值)的所有行都将保存在分区p3中。在将来的某个时候,当商店数已经增长到25, 30, 或更多 ,可以使用ALTER TABLE语句为商店21-25, 26-30,等等增加新的分区。
     在几乎一样的结构中,你还可以基于雇员的工作代码来分割表,也就是说,基于job_code 列值的连续区间。例如——假定2位数字的工作代码用来表示普通(店内的)工人,三个数字代码表示办公室和支持人员,四个数字代码表示管理层,你可以使用下面的语句创建该分区表:
Sql代码
  1. CREATE TABLE employees (   
  2.     id INT NOT NULL,   
  3.     fname VARCHAR(30),   
  4.     lname VARCHAR(30),   
  5.     hired DATE NOT NULL DEFAULT '1970-01-01',   
  6.     separated DATE NOT NULL DEFAULT '9999-12-31',   
  7.     job_code INT NOT NULL,   
  8.     store_id INT NOT NULL  
  9. )   
  10.   
  11. PARTITION BY RANGE (job_code) (   
  12.     PARTITION p0 VALUES LESS THAN (100),   
  13.     PARTITION p1 VALUES LESS THAN (1000),   
  14.     PARTITION p2 VALUES LESS THAN (10000)   
  15. );  
 
在这个例子中, 店内工人相关的所有行将保存在分区p0中,办公室和支持人员相关的所有行保存在分区p1中,管理层相关的所有行保存在分区p2中。
       在VALUES LESS THAN 子句中使用一个表达式也是可能的。这里最值得注意的限制是MySQL 必须能够计算表达式的返回值作为LESS THAN (<)比较的一部分;因此,表达式的值不能为NULL 。由于这个原因,雇员表的hired, separated, job_code,和store_id列已经被定义为非空(NOT NULL)。
       除了可以根据商店编号分割表数据外,你还可以使用一个基于两个DATE (日期)中的一个的表达式来分割表数据。例如,假定你想基于每个雇员离开公司的年份来分割表,也就是说,YEAR(separated)的值。实现这种分区模式的CREATE TABLE 语句的一个例子如下所示:
Sql代码
  1. CREATE TABLE employees (   
  2.     id INT NOT NULL,   
  3.     fname VARCHAR(30),   
  4.     lname VARCHAR(30),   
  5.     hired DATE NOT NULL DEFAULT '1970-01-01',   
  6.     separated DATE NOT NULL DEFAULT '9999-12-31',   
  7.     job_code INT,   
  8.     store_id INT  
  9. )   
  10.   
  11. PARTITION BY RANGE (YEAR(separated)) (   
  12.     PARTITION p0 VALUES LESS THAN (1991),   
  13.     PARTITION p1 VALUES LESS THAN (1996),   
  14.     PARTITION p2 VALUES LESS THAN (2001),   
  15.     PARTITION p3 VALUES LESS THAN MAXVALUE   
  16. );  
 在这个方案中,在1991年前雇佣的所有雇员的记录保存在分区p0中,1991年到1995年期间雇佣的所有雇员的记录保存在分区p1中, 1996年到2000年期间雇佣的所有雇员的记录保存在分区p2中,2000年后雇佣的所有工人的信息保存在p3中。RANGE分区在如下场合特别有用:
      1)、 当需要删除一个分区上的“旧的”数据时,只删除分区即可。如果你使用上面最近的那个例子给出的分区方案,你只需简单地使用 “ALTER TABLE employees DROP PARTITION p0;”来删除所有在1991年前就已经停止工作的雇员相对应的所有行。对于有大量行的表,这比运行一个如“DELETE FROM employees WHERE YEAR (separated) <= 1990;”这样的一个DELETE查询要有效得多。
      2)、想要使用一个包含有日期或时间值,或包含有从一些其他级数开始增长的值的列。
      3)、经常运行直接依赖于用于分割表的列的查询。例如,当执行一个如“SELECT COUNT(*) FROM employees WHERE YEAR(separated) = 2000 GROUP BY store_id;”这样的查询时,MySQL可以很迅速地确定只有分区p2需要扫描,这是因为余下的分区不可能包含有符合该WHERE子句的任何记录。注释:这种优化还没有在MySQL 5.1源程序中启用,但是,有关工作正在进行中。
  • LIST分区
      类似于按RANGE分区,区别在于LIST分区是基于列值匹配一个离散值集合中的某个值来进行选择。      LIST分区通过使用“PARTITION BY LIST(expr)”来实现,其中“expr” 是某列值或一个基于某个列值、并返回一个整数值的表达式,然后通过“VALUES IN (value_list)”的方式来定义每个分区,其中“value_list”是一个通过逗号分隔的整数列表
注释:在MySQL 5.1中,当使用LIST分区时,有可能只能匹配整数列表。
Sql代码
  1. CREATE TABLE employees (   
  2.     id INT NOT NULL,   
  3.     fname VARCHAR(30),   
  4.     lname VARCHAR(30),   
  5.     hired DATE NOT NULL DEFAULT '1970-01-01',   
  6.     separated DATE NOT NULL DEFAULT '9999-12-31',   
  7.     job_code INT,   
  8.     store_id INT  
  9. );  
 
假定有20个音像店,分布在4个有经销权的地区,如下表所示:====================
地区      商店ID 号------------------------------------北区      3, 5, 6, 9, 17
东区      1, 2, 10, 11, 19, 20
西区      4, 12, 13, 14, 18
中心区   7, 8, 15, 16====================
要按照属于同一个地区商店的行保存在同一个分区中的方式来分割表,可以使用下面的“CREATE TABLE”语句:
Sql代码
  1. CREATE TABLE employees (   
  2.     id INT NOT NULL,   
  3.     fname VARCHAR(30),   
  4.     lname VARCHAR(30),   
  5.     hired DATE NOT NULL DEFAULT '1970-01-01',   
  6.     separated DATE NOT NULL DEFAULT '9999-12-31',   
  7.     job_code INT,   
  8.     store_id INT  
  9. )   
  10.   
  11. PARTITION BY LIST(store_id)   
  12.     PARTITION pNorth VALUES IN (3,5,6,9,17),   
  13.     PARTITION pEast VALUES IN (1,2,10,11,19,20),   
  14.     PARTITION pWest VALUES IN (4,12,13,14,18),   
  15.     PARTITION pCentral VALUES IN (7,8,15,16)   
  16. );  
 这使得在表中增加或删除指定地区的雇员记录变得容易起来。例如,假定西区的所有音像店都卖给了其他公司。那么与在西区音像店工作雇员相关的所有记录(行)可以使用查询“ALTER TABLE employees DROP PARTITION pWest;”来进行删除,它与具有同样作用的DELETE (删除)查询“DELETE query DELETE FROM employees WHERE store_id IN (4,12,13,14,18);”比起来,要有效得多。
【要点】:如果试图插入列值(或分区表达式的返回值)不在分区值列表中的一行时,那么“INSERT”查询将失败并报错。例如,假定LIST分区的采用上面的方案,下面的查询将失败:
Sql代码
  1. INSERT INTO employees VALUES(224, 'Linus''Torvalds''2002-05-01''2004-10-12', 42, 21);  
 
这是因为“store_id”列值21不能在用于定义分区pNorth, pEast, pWest,或pCentral的值列表中找到。要重点注意的是,LIST分区没有类似如“VALUES LESS THAN MAXVALUE”这样的包含其他值在内的定义。将要匹配的任何值都必须在值列表中找到。LIST分区除了能和RANGE分区结合起来生成一个复合的子分区,与HASH和KEY分区结合起来生成复合的子分区也是可能的。
  • HASH分区         
       基于用户定义的表达式的返回值来进行选择的分区,该表达式使用将要插入到表中的这些行的列值进行计算。这个函数可以包含MySQL 中有效的、产生非负整数值的任何表达式。      要使用HASH分区来分割一个表,要在CREATE TABLE 语句上添加一个“PARTITION BY HASH (expr)”子句,其中“expr”是一个返回一个整数的表达式。它可以仅仅是字段类型为MySQL 整型的一列的名字。此外,你很可能需要在后面再添加一个“PARTITIONS num”子句,其中num 是一个非负的整数,它表示表将要被分割成分区的数量。
Sql代码
  1. CREATE TABLE employees (   
  2.     id INT NOT NULL,   
  3.     fname VARCHAR(30),   
  4.     lname VARCHAR(30),   
  5.     hired DATE NOT NULL DEFAULT '1970-01-01',   
  6.     separated DATE NOT NULL DEFAULT '9999-12-31',   
  7.     job_code INT,   
  8.     store_id INT  
  9. )   
  10. PARTITION BY HASH(store_id)   
  11. PARTITIONS 4;  
 如果没有包括一个PARTITIONS子句,那么分区的数量将默认为1。 例外: 对于NDB Cluster(簇)表,默认的分区数量将与簇数据节点的数量相同,这种修正可能是考虑任何MAX_ROWS 设置,以便确保所有的行都能合适地插入到分区中。
  • LINER HASH
MySQL还支持线性哈希功能,它与常规哈希的区别在于,线性哈希功能使用的一个线性的2的幂(powers-of-two)运算法则,而常规 哈希使用的是求哈希函数值的模数。
线性哈希分区和常规哈希分区在语法上的唯一区别在于,在“PARTITION BY” 子句中添加“LINEAR”关键字。
Sql代码
  1. CREATE TABLE employees (   
  2.     id INT NOT NULL,   
  3.     fname VARCHAR(30),   
  4.     lname VARCHAR(30),   
  5.     hired DATE NOT NULL DEFAULT '1970-01-01',   
  6.     separated DATE NOT NULL DEFAULT '9999-12-31',   
  7.     job_code INT,   
  8.     store_id INT  
  9. )   
  10. PARTITION BY LINEAR HASH(YEAR(hired))   
  11. PARTITIONS 4;  
 假设一个表达式expr, 当使用线性哈希功能时,记录将要保存到的分区是num 个分区中的分区N,其中N是根据下面的算法得到:
1.    找到下一个大于num.的、2的幂,我们把这个值称为V ,它可以通过下面的公式得到:
2.    V = POWER(2, CEILING(LOG(2, num)))
(例如,假定num是13。那么LOG(2,13)就是3.7004397181411。 CEILING(3.7004397181411)就是4,则V = POWER(2,4), 即等于16)。
3.    设置 N = F(column_list) & (V - 1).
4.    当 N >= num:
·         设置 V = CEIL(V / 2)
·         设置 N = N & (V - 1)
例如,假设表t1,使用线性哈希分区且有4个分区,是通过下面的语句创建的:
CREATE TABLE t1 (col1 INT, col2 CHAR(5), col3 DATE)
    PARTITION BY LINEAR HASH( YEAR(col3) )
    PARTITIONS 6;
现在假设要插入两行记录到表t1中,其中一条记录col3列值为'2003-04-14',另一条记录col3列值为'1998-10-19'。第一条记录将要保存到的分区确定如下:
V = POWER(2, CEILING(LOG(2,7))) = 8
N = YEAR('2003-04-14') & (8 - 1)
   = 2003 & 7
   = 3
(3 >= 6 为假(FALSE): 记录将被保存到#3号分区中)
第二条记录将要保存到的分区序号计算如下:
V = 8
N = YEAR('1998-10-19') & (8-1)
  = 1998 & 7
  = 6
(6 >= 4 为真(TRUE): 还需要附加的步骤
N = 6 & CEILING(5 / 2)
  = 6 & 3
  = 2
 
(2 >= 4 为假(FALSE): 记录将被保存到#2分区中)
按照线性哈希分区的优点在于增加、删除、合并和拆分分区将变得更加快捷,有利于处理含有极其大量(1000吉)数据的表。它的缺点在于,与使用常规HASH分区得到的数据分布相比,各个分区间数据的分布不大可能均衡。
  • KSY分区
类似于按HASH分区,区别在于KEY分区只支持计算一列或多列,且MySQL 服务器提供其自身的哈希函数。必须有一列或多列包含整数值。
Sql代码 

CREATE TABLE tk (   
  1.     col1 INT NOT NULL,   
  2.     col2 CHAR(5),   
  3.     col3 DATE  
  4. )   
  5. PARTITION BY LINEAR KEY (col1)   
  6. PARTITIONS 3;  
 在KEY分区中使用关键字LINEAR和在HASH分区中使用具有同样的作用,分区的编号是通过2的幂(powers-of-two)算法得到,而不是通过模数算法。
分享到:
评论

相关推荐

    阿里巴巴Cobar架构设计与实践+数据库优化等4篇

    以下是对Cobar架构及数据库优化的深入解析: 1. Cobar架构概述: Cobar的核心功能是作为数据库连接池,将来自应用层的SQL请求分发到后端的多个MySQL实例,实现负载均衡。它采用了基于内存的数据处理方式,大大提升...

    表分区资料表分区资料

    根据给定的信息,我们可以推断出这是一篇与数据库中的“表分区”技术相关的文章。由于原文中的部分内容难以理解,我们将主要依据标题、描述和标签来进行详细的解释。 ### 表分区资料 #### 一、什么是表分区? 表...

    Oracle数据库优化方案.pdf

    在这篇文章中,我们将讨论Oracle数据库优化方案,包括内存等参数配置的优化、减少物理读写的优化、批量重复操作的SQL语句及大表操作的优化等。同时,我们还将讨论Oracle数据库优化的前言、概述、方案、前后比较和...

    大型ORACLE数据库优化设计方案

    在《大型ORACLE数据库优化设计方案》这篇论文中,作者深入探讨了针对ORACLE数据库的多层次优化方法,旨在提升数据库的性能和效率。以下是关于这个主题的详细说明: 一、数据库优化自由结构OFA OFA(Optimal ...

    ORACLE数据库优化设计 (2).pdf

    《ORACLE数据库优化设计》这篇文章主要探讨了在ORACLE数据库环境下如何进行性能优化,作者张海霞通过四个不同级别的调整方案进行了深入分析。数据库性能优化包括内存优化、I/O调整、CPU调整、数据库缓存调整、SQL...

    数据库优化最佳实践.pptx

    数据库优化是提升系统性能的关键环节,特别是在大数据量和高并发的场景下,其重要性不言而喻。本文件"数据库优化最佳实践.pptx"主要涵盖了优化的思路、实践案例以及总结,以下是对这些内容的详细解读。 一、介绍 在...

    3篇Oracle数据库性能优化的论文

    最后,“Oracle数据库性能调优技术与实现”可能会提供更广泛的性能优化方法,包括物理设计优化(如表分区、索引设计)、内存管理(如PGA和SGA的调整)、并行执行策略、以及数据库维护操作的最佳实践。论文可能还会...

    数据库优化

    本篇文章将深入探讨数据库优化的相关技术,旨在帮助读者掌握提高数据库效率的关键策略。 一、索引优化 索引是数据库中提升查询速度的重要工具。合理创建和使用索引能显著减少数据检索时间。应根据查询语句的WHERE...

    宝钢OA系统Oracle数据库优化技术的研究与应用.zip

    《宝钢OA系统Oracle数据库优化技术的研究与应用》是一篇深度探讨企业信息系统中数据库优化重要性的技术案例。本文主要关注的是在宝钢的办公自动化(OA)系统中,如何通过有效的Oracle数据库优化策略来提升系统性能,...

    秒杀应用的MySQL数据库优化 (转载)

    秒杀应用的MySQL数据库优化是一个重要的议题,尤其是在高并发、数据处理速度要求极高的场景下。这类应用常常面临巨大的压力,如瞬间涌入的大量请求、数据读写速度、以及资源的有效利用。本篇文章将深入探讨如何针对...

    DB2数据库性能调整和优化

    本篇文章将深入探讨DB2数据库性能调整与优化的核心概念、方法以及实践技巧。 首先,理解数据库性能的基础是了解SQL查询执行的原理。在DB2中,查询优化器负责分析SQL语句并选择最佳执行计划。优化器的工作包括解析...

    简单了解MYSQL数据库优化阶段

    你们是如何进行数据库优化的? 那这个问题应该怎么答呢?其实写这个题材的原因是我这几天看到各公众号转的一篇数据库调优的知识(不上链接了),我就稍微翻了几下,上面动不动就来说要对数据库进行水平拆分,我就想...

    SQL Server数据库性能优化的几点分析.pdf

    这三方面的内容组成了数据库优化的主要部分,并对当前网络技术发展下,数据库应用系统中数据量的增加以及对数据库性能优化的需求进行了深入分析。 一、有效的利用索引 在数据库的查询优化中,索引起到了非常关键的...

    数据库优化设计之三十六计

    ### 数据库优化设计之三十六计 #### 数据库实例篇 **第一计:数据库内存占系统内存2/3大小** 为了最大化数据库性能并确保系统的稳定性,通常建议将系统内存的约2/3分配给数据库使用。这意味着需要为数据库预留...

    网络数据库的优化毕业设计论文

    【网络数据库的优化毕业设计论文】探讨了在网络环境中如何...综上所述,这篇毕业设计论文深入研究了网络数据库的性能优化,通过实践应用Oracle 9i的高级特性,为数据库管理员和开发人员提供了有价值的理论和实践经验。

    毕业论文-数据库查询优化策略分析

    本文档是一篇关于“数据库查询优化策略分析”的毕业论文,主要研究如何提升数据库查询效率,内容包括基于索引的优化策略和SQL语句的优化方法。随着信息技术的发展,数据库在各行各业中的重要性日益凸显,数据库查询...

    SYBASE数据库优化

    本篇文章将深入探讨Sybase数据库优化的关键技术和策略,以提高其使用效率。 1. **索引优化**: 索引是提升查询速度的关键。创建合适的索引可以大幅减少数据检索时间。应关注高频率查询的列,避免在更新频繁的列上...

    用分区技术改造Oracle数据库的方法探索与实践.pdf

    本篇文章主要探讨了利用分区技术改造Oracle数据库的方法,特别是针对某快递公司合并后业务数据急剧增加的实际情况,提出了一系列改造方案和实践操作。 首先,文章的前言部分简要介绍了中国快递行业的发展现状以及...

Global site tag (gtag.js) - Google Analytics