会员登录 - 用户注册 - 设为首页 - 加入收藏 - 网站地图 MySQL 保存日期,用哪种数据类型合适?datetime?timestamp?还是 int?!

MySQL 保存日期,用哪种数据类型合适?datetime?timestamp?还是 int?

时间:2025-11-05 10:41:36 来源:益强数据堂 作者:人工智能 阅读:840次

日期算是存日我们在日常开发中经常用到的数据类型,一般来说一张表都有 createTime 和 updateTime 字段,期用MySQL 中针对日期也提供了很多种不同的种数数据类型,如:

datetimetimestampint

等等。据类甚至也有人直接将日期存为字符串的型合。

那么到底该用哪种类型来保存日期呢?存日

1. 字符串

在这些类型中,首先应该排除掉的期用就是字符串了,很多新手小伙伴爱用字符串存储日期,种数但实际上这并不是据类一个很好的方案。

使用字符串存储日期,型合第一个显而易见的存日问题就是无法使用 MySQL 中提供的日期函数,这会为很多查询带来不便。期用

例如用户表中有一个字段 birthday,种数表示用户的据类生日,现在想要查询 2001 出生的型合所有用户,如果 birthday 是 日期类型,就可以使用 YEAR 函数,但是如果 birthday 是字符串类型,免费信息发布网这个问题就不太好处理了。

使用字符串存储日期的第二个问题就是占用空间较大,例如存储如下时间:

2021-01-01 00:00:00

如果使用字符串,需要 19 个字节。如果使用 datetime 需要 8 个字节。如果使用 timestamp 需要 4 个字节。

所以首先排除掉字符串。

2. DATETIME VS TIMESTAMEP

2.1 占用空间

DATETIME 在数据库中存储的形式为:YYYY-MM-DD hh:mm:ss,至于占用的字节数,则看情况,我们来看一段来自 MySQL 官网的内容:

可以看到,MySQL5.6.4 是一个分水岭:

在 MySQL5.6.4 之前,DATETIME 固定占用 8 个字节。从 MySQL5.6.4 开始,DATETIME 类型开始支持毫秒,DATETIME(N) 中的 N 表示毫秒的精度,例如,DATETIME(6) 表示可以存储 6 位的毫秒值,那么此时,DATETIME 占用的字节数,就跟后面的毫秒数有关了,如果 DATETIME 没有详细到毫秒,站群服务器那么占用 5 个字节,如果详细到毫秒了,那就看情况,根据毫秒的精度,占用不同的空间,毫秒精度小于等于 2 时,总共占用 6 个字节;毫秒精度小于等于 4 时,总共占用 7 个字节;毫秒精度小于等于 6 时,总共占用 8 个字节。

同样,由上图我们也可以看出,在 MySQL5.6.4 之前,TIMESTAMEP 固定占用 4 个字节,从 MySQL5.6.4 开始,依据毫秒的精度,TIMESTAMEP 占用的字节数介于 4 到 7 之间。

所以无论是 TIMESTAMEP,还是 DATETIME,都是比字符串节省空间的。

2.2 存储范围

DATETIME 的存储范围介于 1000-01-01 00:00:00 到 9999-12-31 23:59:59 之间。

TIMESTAMP 的存储范围则介于 1970-01-01 00:00:01 UTC 到 2038-01-19 03:14:07 UTC 之间。

很明显 DATETIME 的存储范围要更大一些。源码库

2.3 底层存储TIMESTAMP 类型最大的优势在于自带时区属性,因为它本质上是从毫秒转化而来。如果你的业务需要对应不同的国家时区,那么类型 TIMESTAMP 是一种不错的选择,TIMESTAMP 类型字段的值会随着服务器时区的变化而变化,自动换算成相应的时间,说简单点就是在不同时区,查询到同一个条记录此字段的值会不一样。

举个 TIMESTAMP 的使用场景例子:

新闻类的业务,通常用户想知道这篇新闻发布时对应的自己国家时间,那么 TIMESTAMP 是一种不错的选择。

TIMESTAMP 会随着时区的变化而自动调整,而 DATETIME 不会。

我举个例子:假设我数据库目前的时区是 Asia/Shanghai:

现在有一个 user 表,数据如下:

其中,createTime 字段是 DATETIME,而 updateTime 是 TIMESTAMP,现在我修改一下数据库时区,我们再来查看:

小伙伴们可以看到,我把时区设置为东京,东京比我们快一个小时,此时 updateTime 自动变了,而 DATETIME 不变。

时区的问题一定要谨慎,不过时区问题也并非一定要在数据库中解决,也可以在前端或者服务端用代码处理下。

2.4 性能比较

从毫秒数转换到 TIMESTAMP 并不费事,但是当要进行时区转换的时候,需要调用操作系统底层系统函数,而这个函数需要额外的加锁操作,以确保这时操作系统时区没有修改,一加锁,效率就低了。

对于这个问题,只存在于 TIMESTAMP 中,因为 DATETIME 不存在时区转化问题。

对于 TIMESTAMP,建议使用显式的时区,而不是操作系统时区。

3. int

字符串费空间,TIMESTAMP 和 DATETIME 如果没有吃透则总感觉乱乱的,所以也有人存时间戳,存一个 int 类型的数值,用一个时间戳来表示时间。

用 int 保存时间的话,当我们需要进行日期排序以及按照日期范围查询的时候,就变成了普通的数字比较了,那么效率肯定是杠杠滴。

不过 int 有一个致命的问题就是可读性太差,所以用不用 int 就要仔细斟酌看情况了。

好啦,小伙伴们留言说说你日常开发日期用的哪种类型呢?出于什么样的考虑用了该类型?

(责任编辑:系统运维)

上一篇:华硕玩家国度M9C(华硕M9C的性能表现、游戏特色及用户评价)
下一篇:一早起来打开电脑,无法启动显示 no such device blablablagrub rescue>灰常的茫然,无奈,只得开启家中的那台老家伙。一顿baidu google 后,大概清楚是因为昨天晚上的自动升级导致,grub加载软件出了些问题。网上有两种解决方法:本人尝试一种:成功了! 由于这个问题相当普遍,所以写个小报告,为后人指点迷津~~~本人电脑装有xp 和 ubuntu10.04 。问题描述:开机初使化过程中出现的grub错误,原因可能多种,我这是由于升级导致的原因: 主引导记录MBR消失了。解决方法:一种是在windows中修复MBR(mbtfix。Exe)另一种是在ubuntu中修复MBR普及下知识MBR:The MBR may be used for one or more of the following:Holding a disks primary partition table.Bootstrapping operating systems, after the computers BIOS passes execution to machine code instructions contained within the MBR.Uniquely identifying individual disk media, with a 32-bit disk signature; even though it may never be used by the machine the disk is running on.我是通过第二种方法:自行制作一个livecd,一种可以自启动的linux 安装盘 。我没法烧cd,所以通过U盘制作了一个启动盘: 网址: http://www.ubuntu.com/desktop/get-ubuntu/download上面有很详细的步骤。1.首先下载一个ubuntu的iso镜像文件2.下载universal usb installer3.运行installer 將u盘制作成启动盘(移动硬盘不行)4设置电脑的加载顺序,“先启动usb设备”。重新启动5.直接启动ubuntu,打开terminal。输入 :sudo apt-get install lilosudo lilo -M /dev/sda mbr(忽略所有警告,我们只要mbr)6,重启,修改加载顺序,一切正常~~~
推荐内容
  • 三星A8升级后的全新体验(全方位升级,让你爱不释手)
  • Win7无线网卡使用教程(一步一步教你如何在Win7系统下使用无线网卡连接网络)
  • 完全指南(一步一步教你用ISO文件重新安装电脑系统,让电脑焕然一新)
  • Win10Ghost重装教程
  • 戴尔7平板电脑(轻薄便携、高性能全能型平板电脑)
  • 用U盘安装ISO系统的教程(轻松安装ISO系统,告别繁琐步骤)
热点内容