平时开发中经常需要记录时间,比如用于记录某条记录的创建时间以及修改时间。在数据库中存储时间的方式有很多种,比如 MySQL 本身就提供了日期类型,比如 DATETIME,TIMESTAMEP 等,我们也可以直接存储时间戳为 INT 类型,也有人直接将时间存储为字符串类型。

那么到底哪种存储时间的方式更好呢?

不要使用字符串存储时间类型

这是初学者很容易犯的错误,容易直接将字段设置为 VARCHAR 类型,存储"2021-01-01 00:00:00"这样的字符串。当然这样做的优点是比较简单,上手快。

但是极力不推荐这样做,因为这样做有两个比较大的问题:

  • 字符串占用的空间大

  • 这样存储的字段比较效率太低,只能逐个字符比较,无法使用 MySQL 提供的日期API

MySQL 中的日期类型

MySQL 数据库中常见的日期类型有 YEAR、DATE、TIME、DATETIME、TIMESTAMEP。因为一般都需要将日期精确到秒,其中比较合适的有DATETIME,TIMESTAMEP。

DATETIME

DATETIME 在数据库中存储的形式为:YYYY-MM-DD HH:MM:SS,固定占用 8 个字节。

从 MySQL 5.6 版本开始,DATETIME 类型支持毫秒,DATETIME(N) 中的 N 表示毫秒的精度。例如,DATETIME(6) 表示可以存储 6 位的毫秒值。

TIMESTAMEP

TIMESTAMP 实际存储的内容为‘1970-01-01 00:00:00’到现在的毫秒数。在 MySQL 中,由于类型 TIMESTAMP 占用 4 个字节,因此其存储的时间上限只能到‘2038-01-19 03:14:07’。

从 MySQL 5.6 版本开始,类型 TIMESTAMP 也能支持毫秒。与 DATETIME 不同的是,若带有毫秒时,类型 TIMESTAMP 占用 7 个字节,而 DATETIME 无论是否存储毫秒信息,都占用 8 个字节。

类型 TIMESTAMP 最大的优点是可以带有时区属性,因为它本质上是从毫秒转化而来。如果你的业务需要对应不同的国家时区,那么类型 TIMESTAMP 是一种不错的选择。比如新闻类的业务,通常用户想知道这篇新闻发布时对应的自己国家时间,那么 TIMESTAMP 是一种选择。Timestamp 类型字段的值会随着服务器时区的变化而变化,自动换算成相应的时间,说简单点就是在不同时区,查询到同一个条记录此字段的值会不一样。

TIMESTAMP 的性能问题

TIMESTAMP 还存在潜在的性能问题。

虽然从毫秒数转换到类型 TIMESTAMP 本身需要的 CPU 指令并不多,这并不会带来直接的性能问题。但是如果使用默认的操作系统时区,则每次通过时区计算时间时,要调用操作系统底层系统函数 __tz_convert(),而这个函数需要额外的加锁操作,以确保这时操作系统时区没有修改。所以,当大规模并发访问时,由于热点资源竞争,会产生两个问题:

  • 性能不如 DATETIME:DATETIME 不存在时区转化问题。

  • 性能抖动:海量并发时,存在性能抖动问题。

为了优化 TIMESTAMP 的使用,建议使用显式的时区,而不是操作系统时区。比如在配置文件中显示地设置时区,而不要使用系统时区:

[mysqld]time_zone = "+08:00"

简单总结一下这两种数据类型的优缺点:

  • DATETIME 没有存储的时间上限,而TIMESTAMP存储的时间上限只能到‘2038-01-19 03:14:07’

  • DATETIME 不带时区属性,需要前端或者服务端处理,但是仅从数据库保存数据和读取数据而言,性能更好

  • TIMESTAMP 带有时区属性,但是每次需要通过时区计算时间,并发访问时会有性能问题

  • 存储 DATETIME 比 TIMESTAMEP 多占用一部分空间

数值型时间戳(INT)

很多时候,我们也会使用 int 或者 bigint 类型的数值也就是时间戳来表示时间。

这种存储方式的具有 Timestamp 类型的所具有一些优点,并且使用它的进行日期排序以及对比等操作的效率会更高,跨系统也很方便,毕竟只是存放的数值。缺点也很明显,就是数据的可读性太差了,你无法直观的看到具体时间。

如果需要查看某个时间段内的数据

select * from t where created_at > UNIX_TIMESTAMP('2021-01-01 00:00:00');

DATETIME vs TIMESTAMP vs INT,怎么选?

每种方式都有各自的优势,下面再对这三种方式做一个简单的对比:

TIMESTAMP 与 INT 本质一样,但是相比而言虽然 INT 对开发友好,但是对 DBA 以及数据分析人员不友好,可读性差。所以《高性能 MySQL 》的作者推荐 TIMESTAMP 的原因就是它的数值表示时间更加直观。下面是原文:

至于时区问题,可以由前端或者服务这里做一次转化,不一定非要在数据库中解决。

总结

本文比较了几种最常使用的存储时间的方式,我最推荐的还是 DATETIME。理由如下:

  • TIMESTAMP 比数值型时间戳可读性更好

  • DATETIME 的存储上限为 9999-12-31 23:59:59,如果使用 TIMESTAMP,则 2038 年需要考虑解决方案

  • DATETIME 由于不需要时区转换,所以性能比 TIMESTAMP 好

  • 如果需要将时间存储到毫秒,TIMESTAMP 要 7 个字节,和 DATETIME 8 字节差不太多

MySQL 中存储时间的最佳实践相关推荐

  1. Zeno节点系统中的C++最佳实践

    文章目录 1.经典的多态案例 (1)多态用于设计模式之"模板模式" (2)shared_ptr 如何深拷贝? (3)能把拷贝构造函数也作为虚函数? 5.CRTP 6.类型擦除 7. ...

  2. #今日论文推荐# 爱丁堡大学等首篇《移动无线网络中的深度学习》综述论文,67页pdf涵盖570篇文献阐述深度学习在移动无线网络中的应用最佳实践

    #今日论文推荐# 爱丁堡大学等首篇<移动无线网络中的深度学习>综述论文,67页pdf涵盖570篇文献阐述深度学习在移动无线网络中的应用最佳实践 移动设备的迅速普及以及移动应用和服务的日益普 ...

  3. .NET Core 2.1中的HttpClientFactory最佳实践

    ASP.NET Core 2.1中出现一个新的HttpClientFactory功能, 它有助于解决开发人员在使用HttpClient实例从其应用程序发出外部Web请求时可能遇到的一些常见问题. 介绍 ...

  4. 为什么java需要静态类_java – 为什么OOP中静态类的最佳实践有所不同?

    我目前正在阅读有关 Java最佳实践的内容,我发现根据 this book,我们必须支持非静态的静态类.我记得在C#最佳实践中,我们必须根据Dennis Doomen的C#3.0,4.0和5.0编码指 ...

  5. [专栏精选]Unity中的Git最佳实践

    本文节选自洪流学堂公众号技术专栏<大话Unity2018>,未经允许不可转载. 洪流学堂公众号回复专栏,查看更多专栏文章. 小新:"我昨天尝试了一下使用Git来管理Unity项目 ...

  6. Vue中CSS模块化最佳实践

    Vue风格指南中介绍了单文件组件中的Style是必须要有作用域的,否则组件之间可能相互影响,造成难以调试. 在Vue Loader Scope CSS和Vue Loader CSS Modules两节 ...

  7. 测试自动化中遵循的最佳实践

    现在的企业从以往一年.一个月.一周交付,逐渐转变为开始关注两到三天甚至一天内交付.软件自动化测试的出现及其与手动测试的结合使得这种时长的交付成为可能. 及时的软件测试会发现问题,在功能到达用户之前对其 ...

  8. 金仓数据库 MySQL 至 KingbaseES 迁移最佳实践(3. MySQL 数据库移植实战)

    3. MySQL 数据库移植实战 由于 KingbaseES 利用 KDTS-PLUS 等多种工具简化移植过程. 本节重点描述了在实际应用中移植一个 MySQL 数据库系统的完整过程,以及其中的主要移 ...

  9. 京东MySQL数据库Docker化最佳实践(附PPT)

    讲师介绍   刘风才 京东资深数据库专家 2012年加入京东,担任MySQL DBA一职,负责数据库架构设计.数据库性能优化等日常运维工作,参与过分布式数据库项目.多中心交易项目等. 2013~201 ...

最新文章

  1. 一键数据分析自动化特征工程!
  2. 对话Nullmax无人车CEO徐雷:造血营收L3,追梦宏图L4
  3. python求函数曲率_【Python】车道线拟合曲线的曲率半径计算公式及代码
  4. CVPR 2017 SANet:《SANet: Structure-Aware Network for Visual Tracking》论文笔记
  5. vc2010中开始执行不调试灰的_消防水炮调试,客户看到调试流程一刹那就满意
  6. TSP - 状态压缩dp
  7. 兰州交通大学C语言课程设计,兰州交通大学C语言课程设计报告(完整版).doc
  8. mongoose日期 时间 范围查询
  9. 1420C1. Pokémon Army (easy version)
  10. IEEE 解除华为限制;AWS、谷歌云大规模故障;TypeScript 3.5 发布 | 极客头条
  11. 贝格数据携手清华大学“勘探”大数据富矿
  12. 完整版linux下android源码下载、编译、模拟器启动运行
  13. 详解两个栈实现一个队列(python实现——经典面试题)
  14. 第7章第25节:双图排版:两张图片横竖交错叠加排版 [PowerPoint精美幻灯片实战教程]
  15. Python系列17-数据可视化之下载数据
  16. 用 Python 制作各种用途的二维码
  17. 阿里 P9 用 500 多页手册完成双十一高并发秒杀系统,绝了
  18. macOS SwiftUI 进度指示器组件规范之 01 进度指标是什么 Progress Indicators
  19. 离谱的 CSS!从表盘刻度到剪纸艺术
  20. Unix的学习(一)

热门文章

  1. Servlet_urlpartten配置
  2. 设计模式之_Iterator_05
  3. Java8函数式编程(5)--中间操作(源码)
  4. 【OOP】零钱通项目
  5. gtest 测试java_LangTest
  6. (轉貼) ThinkPad鍵盤設計原理和哲學 (NB) (ThinkPad)
  7. 使用注解实现ssh整合
  8. IBM公布Kitura 1.0和Bluemix Runtime for Swift 3
  9. MySQL的insert ignore与replace into不同
  10. 老男孩教育参观云计算公司