SQL Server datetime数据类型设计以及优化误区

数据库 SQL Server
SQL Server datetime数据类型设计以及数据库优化是SQL Server 数据库操作过程中非常重要工作,但是在进行SQL Server datetime数据类型设计以及数据库优化时经常会遇到一些问题,这就会涉及到一些误区,下文中为大家总结出了经常会遇到的,希望对大家能够有所帮助。

导读:SQL Server数据库的发展就离不开数据库的数据类型设计以及数据库的优化,要想做好数据库中数据类型设计以及数据库优化工作,就首先要熟练掌握下文中介绍的数据类型设计以及数据库优化的误区。

场景:在SQL Server 2005中,有一个表TestDatetime,其中Dates这个字段的数据类型是datetime,如果你看到表的记录如下图所示,你最先想到的是什么呢?

 


(图1:数据列表)

你看到这些数据,是不是觉得这样的设计既浪费了存储空间,又使得这个列的索引增大,查询起来更慢,你也想使用一些其它的数据类型来代替这个datetime吧?

其实大家都是这么想的,这个方向是100%正确的,但是在写这篇文章以前,我进入了两个误区:(如果你中了下面的两个误区,那么请你看看这篇文章吧。)

误区一: 把Dates字段的datetime数据类型换成smalldatetime,这样数据就由:‘2009-04-09 00:00:00.000’变为‘2009-04-09 00:00:00’,这个看起来没有减少多少存储空间哦。

误区二:把Dates字段的datetime数据类型换成char(10),这样数据就由:‘2009-04-09 00:00:00.000’变为‘2009-04-09’,这好像能减少很多存储空间哦。

 

分析

在SQL Server 2005版本中保存日期的数据类型只有两种:datetime、smalldatetime,但是在SQL Server 2008版本中新增了一些日期数据类型:time、date、smalldatetime、datetime、datetime2、datetimeoffset,其中的date类型就能满足我们场景中的需求了,如果你幸运的在使用SQL Server 2008的话,那么恭喜你,请使用date数据类型吧。

但是我就比较可悲一点了,在使用SQL Server 2005的前提下,我进入了误区一、误区二。其实这也是因为自己忽略了一下基础性的东西,如果知道不同数据类型的存储空间大小,也许就很轻易的避免这样低级的错误了。

其实你查看表TestDatetime中的Dates字段的时候,看到查询结果中的:“-”、“:”只是用于显示的,并不是真实保存的时候就这样格式的。

datetime占用8个字节,前4个字节存储base date(即1900年1月1日)之前或之后的天数,后4个字节存储午夜后的毫秒数。值范围:1753-01-01 到 9999-12-31。

smalldatetime占用4个字节,前2个字节存储base date(1900年1月1日)之后的天数。后2个字节存储午夜后的分钟数。值范围:1900-01-01 到 2079-06-06。

date占用3个字节,它比smalldatetime的前2个字节多了1字节,所以值的范围更广了。值范围:0001-01-01 到 9999-12-31。

所以,如果你使用char(10)来保存截断的日期,那么你的存储空间反而更大了。

结论: 如果是SQL Server 2005,那么请你使用smalldatetime吧,数据能节约一半,虽然查询的时候看起来没什么改变;如果你是SQL Server 2008,那么请你使用date吧,

虽然3个字节跟4个字节没有多大的差距,但是从设计上和逻辑清晰度上都有很大的提升,而且差距有些时候并不是1个字节的问题,比如当表数据量达到几个亿的时候,还是有差别的,又或者一条记录可能因为差1个字节就刚刚好给8060字节的页瓜分,这些都不容忽视的。

 

<!--[if !supportLists]-->三、<!--[endif]-->测试

下面我们就从数据存储的大小、索引存储的大小、索引使用时候的速度这几个方面进行测试:(这里只测试数据类型:,,数据的内容都是一样的)

<!--[if !supportLists]-->(一) <!--[endif]-->测试前奏:

<!--[if !supportLists]-->1. <!--[endif]-->创建三种数据类型char(10)、datetime、smalldatetime的表;(表结构如下面SQL)

CREATE TABLE [dbo].[TestDatetime](

[Id] [int] IDENTITY(1,1) NOT NULL,

[Dates] [datetime] NULL,

CONSTRAINT [PK_TestDatetime] PRIMARY KEY CLUSTERED

(

[Id] ASC

)WITH (IGNORE_DUP_KEY = OFF) ON [PRIMARY]

) ON [PRIMARY]

<!--[if !supportLists]-->2. <!--[endif]-->插入相同记录到三个表中;(这里插入1210000条记录)

<!--[if !supportLists]-->3. <!--[endif]-->为[Dates]字段创建索引;(在创建索引的时候可以设置填充因子为100%)

<!--[if !supportLists]-->4. <!--[endif]-->查看索引属性中的索引碎片信息,查看表数据和索引占用的空间,测试[Dates]字段索引的查询效率;

<!--[if !supportLists]-->(二) <!--[endif]-->测试结果:

<!--[if !supportLists]-->1. <!--[endif]-->数据存储大小:

 


(图2:数据空间对比)

索引存储信息:

 

(图3:char(10))

 

(图4:datetime)

 

(图5:smalldatetime)

索引查询的情况:

多次执行,SQL Server执行时间为:[char(10)] 大部分在43~59徘徊,偶尔出现小于10的;[datetime]平均在1~2毫秒;[smalldatetime]均在1毫秒;而且大家会发现[smalldatetime]有其它的9次逻辑读取变为8次了。

 

--[TestChar10]

SQL Server 分析和编译时间:

CPU 时间= 0 毫秒,占用时间= 1 毫秒。

 

(2200 行受影响)

表'TestChar10'。扫描计数1,逻辑读取9 次,物理读取0 次,预读0 次,lob 逻辑读取0 次,lob 物理读取0 次,lob 预读0 次。

 

SQL Server 执行时间:

CPU 时间= 0 毫秒,占用时间= 59 毫秒。

 

SQL Server 执行时间:

CPU 时间= 0 毫秒,占用时间= 1 毫秒。

 

--[TestDatetime]

SQL Server 分析和编译时间:

CPU 时间= 0 毫秒,占用时间= 1 毫秒。

 

(2200 行受影响)

表'TestDatetime'。扫描计数1,逻辑读取9 次,物理读取0 次,预读0 次,lob 逻辑读取0 次,lob 物理读取0 次,lob 预读0 次。

 

SQL Server 执行时间:

CPU 时间= 0 毫秒,占用时间= 2 毫秒。

 

SQL Server 执行时间:

CPU 时间= 0 毫秒,占用时间= 1 毫秒。

 

--[TestSmalldatetime]

SQL Server 分析和编译时间:

CPU 时间= 0 毫秒,占用时间= 1 毫秒。

 

(2200 行受影响)

表'TestSmalldatetime'。扫描计数1,逻辑读取8 次,物理读取0 次,预读0 次,lob 逻辑读取0 次,lob 物理读取0 次,lob 预读0 次。

 

SQL Server 执行时间:

CPU 时间= 0 毫秒,占用时间= 1 毫秒。

 

SQL Server 执行时间:

CPU 时间= 0 毫秒,占用时间= 1 毫秒。

 

--SQL Server 2008新数据类型

SELECT

CAST('2007-05-08 12:35:29. 1234567 +12:15' AS time(7)) AS 'time'

,CAST('2007-05-08 12:35:29. 1234567 +12:15' AS date) AS 'date'

,CAST('2007-05-08 12:35:29.123' AS smalldatetime) AS

'smalldatetime'

,CAST('2007-05-08 12:35:29.123' AS datetime) AS 'datetime'

,CAST('2007-05-08 12:35:29. 1234567 +12:15' AS datetime2(7)) AS 'datetime2'

,CAST('2007-05-08 12:35:29.1234567 +12:15' AS datetimeoffset(7)) AS 'datetimeoffset';

SQL Server datetime数据类型设计以及数据库优化误区就为大家总结这么多,可能还不够全面,以后有机会还会继续为大家介绍更多的总结,希望大家都能够从中有所收获。

【编辑推荐】

  1. SQL Server 2005 自动化删除表分区设计方案
  2. SQL Server数据库中对图片进行保存和输出
  3. 使用SQL Server 2008导入平面文件
  4. SQL Server 2008内存性能监控
责任编辑:迎迎 来源: 博客园
相关推荐

2011-05-11 10:39:01

SQL Serverdatetime数据类优化误区

2010-11-29 10:09:26

SQL Server

2010-07-12 15:36:45

SQL Server

2010-06-28 14:30:08

SQL Server

2010-07-22 17:57:40

2010-09-06 16:25:46

SQL函数

2010-09-25 09:45:46

sql server主

2010-09-13 10:21:26

sql server2

2010-10-19 15:54:38

sql server创

2010-07-26 14:21:25

SQL Server数

2010-07-22 17:47:32

SQL Server数

2010-07-20 15:54:02

SQL Server数

2010-09-13 09:58:17

SQL Server2

2010-07-09 15:10:11

MS SQL Serv

2010-07-23 14:18:47

SQL Server数

2009-04-16 17:55:55

SQL Server 空间数据.NET

2011-08-25 16:31:36

SQL Servertimestamp

2010-06-30 11:31:55

SQL Server数

2010-06-28 11:22:14

MS SQL Serv

2010-07-22 17:39:44

点赞
收藏

51CTO技术栈公众号