Redian新闻
>
面试官:数据库日期类型字段,需要兼容不同数据库,应该如何选择?

面试官:数据库日期类型字段,需要兼容不同数据库,应该如何选择?

公众号新闻

点击上方“芋道源码”,选择“设为星标

管她前浪,还是后浪?

能浪的浪,才是好浪!

每天 10:33 更新文章,每天掉亿点点头发...

源码精品专栏

 
来源:zhangxin.blog.csdn.net/
article/details/124955363

当设计一个产品,其中很多地方要把日期类型保存到数据库中,如果产品有兼容不同数据库产品的需求,那么,应当怎样设计呢?

当然,首先想到的是,使用数据库的 Date 或 DateTime 类型,可是看看不同数据库这些类型间的区别吧,真让人望而止步。

MySQL 数据库:

它们分别是 date、datetime、time、timestamp 和 year。

  • date :“yyyy-mm-dd”格式表示的日期值
  • time :“hh:mm:ss”格式表示的时间值
  • datetime:“yyyy-mm-dd hh:mm:ss”格式
  • timestamp:“yyyymmddhhmmss”格式表示的时间戳值
  • year:“yyyy”格式的年份值。

范围:

  • date “1000-01-01” 到 “9999-12-31” 3字节
  • time “-838:59:59” 到 “838:59:59” 3字节
  • datetime “1000-01-01 00:00:00” 到 “9999-12-31 23:59:59” 8字节
  • timestamp 19700101000000 到 2037 年的某个时刻 4字节
  • year 1901 到 2155 1 字节

基于 Spring Boot + MyBatis Plus + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能

  • 项目地址:https://github.com/YunaiV/ruoyi-vue-pro
  • 视频教程:https://doc.iocoder.cn/video/

Oracle 数据库:

Date 类型的内部编码为12

长度:占用7个字节

数据存储的每一位到第七位分别为:世纪,年,月,日,时,分,秒

TIMESTAMP是支持小数秒和时区的日期/时间类型。对秒的精确度更高

TIMESTAMP WITH TIME ZONE 类型是 TIMESTAMP 的子类型,增加了时区支持,占用13字节的存储空间,最后两位用于保存时区信息

INTERVAL 用于表示一段时间或一个时间间隔的方法。在前面有多次提过。INTERVAL有两种类型.

  • YEAR TO MONTH 能存储年或月指定的一个时间段.
  • DATE TO SECOND 存储天,小时,分钟,秒指定的时间段.

基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能

  • 项目地址:https://github.com/YunaiV/yudao-cloud
  • 视频教程:https://doc.iocoder.cn/video/

sql server:

datetime 和 smalldatetime

  • datetime数据类型所占用的存储空间为8个字节,其中前4个字节用于存储1900年1月1日以前或以后的天数,数值分正负,正数表示在此日期之后的日期,负数表示在此日期之前的日期;后4个字节用于存储从此日零时起所指定的时间经过的毫秒数。
  • smalldatetime数据类型使用4个字节存储数据。其中前2个字节存储从基础日期1900年1月1日以来的天数,后两个字节存储此日零时起所指定的时间经过的分钟数。
  • smalldatetime数据类型与datetime数据类型相似,但其日期时间范围较小,从1900年1月1日到2079年6月6日。此数据类型精度较低,只能精确到分钟,其分钟个位为根据秒数四舍五入的值,即以30秒为界四舍五入。

如果没有兼容多种数据库这个要求,我会毫不犹豫的使用数据库的 Date 类型。

因为如果使用 Java 框架产生代码,对数据库中定义为 Date 类型的字段,甚至能在页面上产生出JS的时间选择框,的确能节省很多开发时间。而兼容不同数据库,就希望产品在由一种数据库,迁移到另外一种数据库时,尽可能小的代价,使用了 Date,看来就很困难了。

有一个疑问,不知道目前流行的ORM对这个处理得是不是好?因为工作不怎么涉及这方面,所以不大了解。

在之前的设计开发中,因为有支持多种数据库这种需求,所以首先否定了日期时间这样的类型。

曾经使用过毫秒数(Java 的 System.currentTimeMillis())这种方式,但是选用这个方式,考虑的不是使用起来是否方便或者数据迁移,而是考虑到下面的原因:

Java 取到的毫秒数是对时间点的一种准确描述。定义如下:java.lang.System.currentTimeMillis(),它返回从 UTC 1970 年 1 月 1 日午夜开始经过的毫秒数。

我们可以看到,这个定义,保证了这个时间值能够被后续设计开发的人员正确和准确的理解,能够为所有的应用正确理解,能够在所有时区上正确反映为正常的时间形式。

当时的产品设计是有海外客户的,所以当时的设计,在数据库里保存的,应该是一个“准确的时间”。例如“20120926080000”实际上并没有严格的表示出时间,因为北京时间2012年9月26日8点和格林威治时间2012年9月26日8点显然是不一样的。

虽然我们都是在一个确切的时区里,例如中国都是使用东八区时间,但是需要考虑的是:

  • 有些产品是可能有海外客户的
  • 产品所运行的机器,时区的设置未必都是东八区。

在这种情况下,如果数据库里的时间不准确,会给程序运行带来问题。这种方式最大的缺点在于:

  • 不方便对时间进行分组查询,比如按月统计、按季 统计
  • DBA在维护时,不能直观的根据返回的行结果,看到简单明了的结果(看到的是毫秒数)

使用这种方式的特点是牺牲一点易用性和可理解性(不易于维护和理解),满足了查询结果的直观性和准确性要求,同时最大限度考虑运行效率。为了解决这个问题,我设计了一个辅助的措施,就是建立一个数据库函数来进行时间转换,把毫秒数的时间转为制定时区和格式的时间串,DBA 在维护时可以使用。测试了 Oracle 和 DB2 上,都可以这样。例如之前的查询的时候为:

SELECT username,user_addtime from userinfo

这个查询显示的是毫秒数,使用内置函数后写成:

SELECT username,date2str(user_addtime) from userinfo

这样返回的就是东八区、预先定义好格式的字符串了。

在之后的设计里,还使用过 YYYYMMDDHHmmSST 格式,其中的“T”指时区,加入时区,带来的影响有:

  • 日期时间字段就不能在使用数值来存储了,字符串比数字存储和检索的效率都要低。
  • 应用程序需要加上额外的处理

带来的好处是:

  • 便于 DBA 维护
  • 到什么时候,即便没有看到数据库设计文档,都能看明白并准确理解数据库中一条信息中,这个字段保存到确切信息

使用这种方式的特点是牺牲一点效率,满足了查询结果的直观性和准确性要求。

总结一下,字段类型的选择,还是根据场景的需要来选择,从功能、效率要求、持续开发的要求、维护的要求几个方面综合考虑。



欢迎加入我的知识星球,一起探讨架构,交流源码。加入方式,长按下方二维码噢

已在知识星球更新源码解析如下:

最近更新《芋道 SpringBoot 2.X 入门》系列,已经 101 余篇,覆盖了 MyBatis、Redis、MongoDB、ES、分库分表、读写分离、SpringMVC、Webflux、权限、WebSocket、Dubbo、RabbitMQ、RocketMQ、Kafka、性能测试等等内容。

提供近 3W 行代码的 SpringBoot 示例,以及超 4W 行代码的电商微服务项目。

获取方式:点“在看”,关注公众号并回复 666 领取,更多内容陆续奉上。

文章有帮助的话,在看,转发吧。

谢谢支持哟 (*^__^*)

微信扫码关注该文公众号作者

戳这里提交新闻线索和高质量文章给我们。
相关阅读
面试官:断网了,还能 ping 通 127.0.0.1 吗?高盛面试官:去年收到10万份简历,99%都是Trash今天的午餐和晚餐热点 | 倒计时1个月!现金管理类理财产品新规过渡期即将结束,7日年化收益率跌破2%,金融机构与投资者该如何选择?应激性溃疡预防性用药如何选择?来看指南怎么说投资移民如何着手?项目如何选择?谨慎选择是第一,美国律师给您提下醒!纽约林肯中心 Lincoln Center 欣赏交响乐演出支架、搭桥 or 药物治疗,左主干冠状动脉疾病该如何选择?两种新的疫苗加强剂如何选择? 专家为民众解惑B端产品人跳槽如何选行业/准备面试?大厂面试官告诉你答案65岁以下严重主动脉瓣疾病的治疗,该如何选择?新加坡的幼儿园应该如何选择?看完这篇你就明白了美国买房,House还是Condo,如何选择?面试官:为什么不建议在 MySQL 中使用 UTF-8 ?“理解了‘君子和而不同’,我们才能对不同的观点兼容并包。” ——布莱尔·苏格曼体育单招报名时间已过半,报考院校时该如何选择?所谓好人缘儿选择字段,在B端系统中的4种意义面试官:“给你十个亿,你怎么花?”胰腺巨大假性囊肿的处理方法如何选择?NeurIPS 2022 | 如何度量知识蒸馏中不同数据增强方法的好坏?一种统计学视角面试官:select......for update 会锁表还是锁行?我拴 Q 了!!央视男面试官:“杨澜,你敢不敢穿比基尼出镜?”她的回答,惊艳全场…面试官:MySQL中的 distinct 和 group by 哪个效率更高?​英国永居VS英国入籍,该如何选择?长期收益较高的两大赛道,2023年应该如何选择?人工授精or试管婴儿,应该如何选择?Bloom 美式早午餐香港优才VS香港高才通,该如何选择?面对疲软的市场,Affirm将如何选择?Andy教授解读数据库的2022:大规模数据库投资大幅放缓、区块链数据库仍然是一个愚蠢的想法2022澳洲会计移民现状!即将毕业该如何选择?【猫本留学】程序员面试反问面试官15道题,网友:到底谁在面试?在美国276.监视器升级面试官:电商库存扣减如何设计?如何防止超卖?
logo
联系我们隐私协议©2024 redian.news
Redian新闻
Redian.news刊载任何文章,不代表同意其说法或描述,仅为提供更多信息,也不构成任何建议。文章信息的合法性及真实性由其作者负责,与Redian.news及其运营公司无关。欢迎投稿,如发现稿件侵权,或作者不愿在本网发表文章,请版权拥有者通知本网处理。