前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >YYYY-MM-DD 的黑锅,我们不背!

YYYY-MM-DD 的黑锅,我们不背!

作者头像
挨踢小子部落阁
发布于 2020-02-17 03:28:12
发布于 2020-02-17 03:28:12
55200
代码可运行
举报
运行总次数:0
代码可运行

点击上方“挨踢小子”,“选择关注”OR“设置星标”

执着派,技术流

作者:兔子托尼啊

https://zhuanlan.zhihu.com/p/101150248

写这篇博文是记录下跨年的bug。

去年隔壁组的小伙伴就是计算两个日期之间间隔的天数,因为跨年的原因计算有误。

当时测试组的小姐姐也没有模拟出来这种场景,导致上生产环境直接影响线上的数据。

今天逛技术论论坛正好遇到Java日期的操作bug。

1 yyyy 和 YYYY

别看字,看代码

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
@Test
public void testWeekBasedYear() {
  Calendar calendar = Calendar.getInstance();
  // 2019-12-31
  calendar.set(2019, Calendar.DECEMBER, 31);
  Date strDate1 = calendar.getTime();
  // 2020-01-01
  calendar.set(2020, Calendar.JANUARY, 1);
  Date strDate2 = calendar.getTime();
  // 大写 YYYY
  SimpleDateFormat formatYYYY = new SimpleDateFormat("YYYY/MM/dd");
  System.out.println("2019-12-31 转 YYYY/MM/dd 格式: " + formatYYYY.format(strDate1));
  System.out.println("2020-01-01 转 YYYY/MM/dd 格式: " + formatYYYY.format(strDate2));
  // 小写 YYYY
  SimpleDateFormat formatyyyy = new SimpleDateFormat("yyyy/MM/dd");
  System.out.println("2019-12-31 转 yyyy/MM/dd 格式: " + formatyyyy.format(strDate1));
  System.out.println("2020-01-01 转 yyyy/MM/dd 格式: " + formatyyyy.format(strDate2));
}

输出结果:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
2019-12-31YYYY/MM/dd 格式: 2020/12/31
2020-01-01YYYY/MM/dd 格式: 2020/01/01
2019-12-31 转 yyyy/MM/dd 格式: 2019/12/31
2020-01-01 转 yyyy/MM/dd 格式: 2020/01/01

细心的同学应该发现了2019-12-31YYYY/MM/dd 此刻变成了2020/12/31

??为何呢?

YYYY这么大的能耐,能跑到2020年代去?

我2019年底买的东西,你如果用YYYY来格式化出库日期,我是不是得到2020年底才能收到货?此bug问题挺大的呀!

YYYY 到底是何方妖怪??

Java's DateTimeFormatter pattern "YYYY" gives you the week-based-year, (by default, ISO-8601 standard) the year of the Thursday of that week.

例子:

下面就是用YYYY格式化代码

  • 12/29/2019 将会格式化到2019年 这一周还属于2019年
  • 12/30/2019 将会格式化到2020年 这一周已经属于2020年

看字说话YYYY,week-based year 是 ISO 8601 规定的。

2019-12-31号这一天,安周算年份已经属于2020年了,格式化之后就变成2020年,后面的月份日期不变。

2 dd 和 DD

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
private static void tryit(int Y, int M, int D, String pat) {
  DateTimeFormatter fmt = DateTimeFormatter.ofPattern(pat);
  LocalDate dat = LocalDate.of(Y,M,D);
  String str = fmt.format(dat);
  System.out.printf("Y=%04d M=%02d D=%02d " +
    "formatted with " +
    "\"%s\" -> %s\n",Y,M,D,pat,str);
}
public static void main(String[] args){
  tryit(2020,01,20,"MM/DD/YYYY");
  tryit(2020,01,21,"DD/MM/YYYY");
  tryit(2020,01,22,"YYYY-MM-DD");
  tryit(2020,03,17,"MM/DD/YYYY");
  tryit(2020,03,18,"DD/MM/YYYY");
  tryit(2020,03,19,"YYYY-MM-DD");
}

输出结果:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Y=2020 M=01 D=20 formatted with "MM/DD/YYYY" -> 01/20/2020
Y=2020 M=01 D=21 formatted with "DD/MM/YYYY" -> 21/01/2020
Y=2020 M=01 D=22 formatted with "YYYY-MM-DD" -> 2020-01-22
Y=2020 M=03 D=17 formatted with "MM/DD/YYYY" -> 03/77/2020
Y=2020 M=03 D=18 formatted with "DD/MM/YYYY" -> 78/03/2020
Y=2020 M=03 D=19 formatted with "YYYY-MM-DD" -> 2020-03-79

看到没有?

最后的3个日期都错误了,这里的大写的DD代表的是处于这一年中那一天,不是处于这个月的那一天。

小伙伴们一定要记住了不要犯类似的错误。

3、结论

YYYY和yyyy不一样的,DD和dd也是不一样要切记。

此锅我们不背。

END

推荐一位有走心的coder,致力于打造一款高质量技术流学习社群,他专注于分享Java技术干货,包括面试攻略,开发技巧,架构设计,职场心得等。

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2020-01-18,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 挨踢小子 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
Java日期处理易踩的十个坑
  我们设置了10小时,但运行结果是22点,而不是10点。因为Calendar.HOUR默认是按12小时制处理的,需要使用Calendar.HOUR_OF_DAY,因为它才是按24小时处理的。
不会飞的小鸟
2020/03/29
1.5K0
元旦到了!一个由 "YYYY-MM-dd" 引发的惨案 !
在元旦假期到来之际,我刚好准备出去飘几天,然而在使用一些 App 的时候,竟然被我发现了一个应该是由于前端粗心而导致的 bug,在 2019.12.30 出发,结果 App 上显示的是 2020.12.30(吓得我以为我的订单下错了,此处是不是该把程序员拉去祭天了)。
JavaFish
2022/01/17
2440
元旦到了!一个由 "YYYY-MM-dd" 引发的惨案 !
昨天你用的 YYYY-MM-dd 被 CTO 捶了吗?
下班回家的路上,习惯性打开 群聊 ,不是为了解答问题,而是不想错过任何一个装 x 的机会。这不,就有胖友聊到一个经典的“神坑”:错误使用 YYYY-MM-dd 格式化时间,导致生产翻车。
芋道源码
2021/01/08
5780
昨天你用的 YYYY-MM-dd 被 CTO 捶了吗?
听说又有兄弟因为用YYYY-MM-dd 被锤了...
还记得去年分享过一篇日期格式化使用 YYYY-MM-dd 的潜在问题(链接如下:http://blog.didispace.com/something-about-YYYY-MM-dd/)的文章不? 历史又重演了... 事故现场 我们先来写个单元测试,重现一下这个问题。 测试逻辑: 1、创建两个日期格式化,一个是出问题的YYYY-MM-dd,另一个是正确用法yyyy-MM-dd 2、分别去格式化两个不同的日期:2020年12月26日(周六),2020年12月27日(周日) 具体代码如下: public c
程序猿DD
2023/04/04
3500
听说又有兄弟因为用YYYY-MM-dd 被锤了...
又双叒有兄弟因为 YYYY-MM-dd 被叫去加班了...
今天上海降温到零下了,一点起床的欲望都没,想着直接睡到吃午饭吧。结果看到群里小伙伴吐槽一早被叫起来修Bug,定睛一看,要修的BUG居然又是之前说过很多次的YYYY-MM-dd问题,这个不是之前已经说过很多次了吗(https://blog.didispace.com/YYYY-MM-dd-2020-again/),怎么还有人中招呢
程序猿DD
2021/12/27
3060
【程序猿硬核科普】Java获取指定时间年月日 | 时间戳转换bug你的项目中招了吗 | yyyy和YYYY的区别
最近在知乎看到有篇回答说:跨年导致日期格式YYYY和yyyy导致日期显示Bug的帖子,微信公众号、一些论坛好多中招了,快来看看你的项目里面有没有这个bug吧,哈哈lo(╥﹏╥)o。
浩Coding
2020/02/13
2.3K0
【程序猿硬核科普】Java获取指定时间年月日 | 时间戳转换bug你的项目中招了吗 | yyyy和YYYY的区别
Java 小记 - 时间的处理与探究
时间的处理与日期的格式转换几乎是所有应用的基础职能之一,几乎所有的语言都会为其提供基础类库。作为曾经 .NET 的重度使用者,赖其优雅的语法,特别是可扩展方法这个神级特性的存在,我几乎没有特意关注过这些个基础类库,他们如同空气一般,你呼吸着,却不用感受其所在何处。煽情结束,入坑 Java 后甚烦其时间处理方式,在此做个总结与备忘。
捷义
2018/07/18
7210
一文带你入坑JDK8的新日期时间类 LocalDate、LocalTime、LocalDateTime
参考 https://blog.csdn.net/duan196_118/article/details/111597682 https://blog.csdn.net/qq_24754061/article/details/95500209 https://xijia.blog.csdn.net/article/details/106007147
时间静止不是简史
2023/02/23
4.9K0
一文带你入坑JDK8的新日期时间类 LocalDate、LocalTime、LocalDateTime
官方JDK的BUG?蘑菇签到又出问题啦!
但是,在 2021 年 12月 26 日,蘑菇群里的小伙伴反馈签到成功,但是当日显示未签到,签到积分也正常发放了。
陌溪
2022/01/05
4060
官方JDK的BUG?蘑菇签到又出问题啦!
数据库存储时间你用对了吗?
我们平时在开发中不可避免的要存储时间,比如我们要记录某条数据的创建时间、更新时间等等。数据库中有多种数据类型可以存储时间,那不同数据类型我们要怎么选择?
Java识堂
2021/01/22
2.4K0
数据库存储时间你用对了吗?
还在使用SimpleDateFormat?你的项目崩没?
日常开发中,我们经常需要使用时间相关类,说到时间相关类,想必大家对SimpleDateFormat并不陌生。主要是用它进行时间的格式化输出和解析,挺方便快捷的,但是SimpleDateFormat并不是一个线程安全的类。在多线程情况下,会出现异常,想必有经验的小伙伴也遇到过。下面我们就来分析分析SimpleDateFormat为什么不安全?是怎么引发的?以及多线程下有那些SimpleDateFormat的解决方案?
Java3y
2019/03/07
4700
还在使用SimpleDateFormat?你的项目崩没?
java获取当前日期和时间(各种方法对比)
System.currentTimeMillis()产生一个当前的毫秒,这个毫秒其实就是自1970年1月1日0时起的毫秒数,类型为long; Date:
ha_lydms
2023/08/09
3.3K0
java获取当前日期和时间(各种方法对比)
Java Date 和 Calendar 实例
当前日期:  2012-03-07 2012-03-07 12:30:11 2012-3-7 12:30:11.101 计算周:  -3 3/5/12 12:30 PM 3/10/12 12:30 PM 3/12/12 12:30 PM 3/3/12 12:30 PM 计算月:  2012-03-01 2012-03-31 2012-02-01 2012-02-29 2012-04-01 2012-04-30 计算年:  2012-01-01 2012-12-31 2011-01-01 2011-12-31 2013-01-01 2013-12-31 366 in 2012 计算季度:  2012-3-7 in [ 2012-1 : 2012-3 ] 31 in [ 2012-3-7 ] true 日期格式转换与计算:  Wed Jun 20 00:00:00 CST 2012 Wednesday 2012-06-02 -> 2012-06-12间隔天数:10
阳光岛主
2019/02/19
3.1K0
java calendar 设置小时_Java Calendar类的时间操作[通俗易懂]
Java Calendar 类时间操作,这也许是创建日历和管理最简单的一个方案,示范代码很简单,演示了获取时间,日期时间的累加和累减,以及比较。
全栈程序员站长
2022/08/12
1.6K0
java calendar 设置小时_Java Calendar类的时间操作[通俗易懂]
常用工具类之:DateUtils
import java.text.ParseException; import java.text.ParsePosition; import java.text.SimpleDateFormat; import java.util.Calendar; import java.util.Date; import java.util.GregorianCalendar; import java.util.TimeZone; import org.joda.time.DateTime; /** * <p>
芈亓
2022/06/17
1.3K0
更正《深入理解高并发编程(第1版)》中的一处错误!
最近,有小伙伴看了我写的《深入理解高并发编程(第1版)》或者在 冰河技术 公号看了《高并发之——SimpleDateFormat类的线程安全问题和解决方案》一文,对文中SimpleDateFormat类线程不安全问题的分析产生了疑惑,并留言或者私信我说明了自己对问题的理解和建议。
冰河
2021/03/24
1.7K0
LocalDateTime、Date时间工具类
参考:Date、LocalTime、LocalDate、LocalDate-时间操作工具类_Hatsune_Miku_的博客-CSDN博客
CBeann
2023/12/25
3080
F**K,能不能让我安安稳稳跨个年
在元旦假期到来之际,我刚好准备出去飘几天,然而在使用一些 App 的时候,竟然被我发现了一个应该是由于前端粗心而导致的 bug,在 2021.12.30 触发,结果 App 上显示的是 2022.12.30(吓得我以为我的订单下错了,此处是不是该把程序员拉去祭天了)。
Java旅途
2022/01/17
1750
F**K,能不能让我安安稳稳跨个年
Java SimpleDateFormat进行日期格式化
众所周知,Java中的日期类是Date,然后日期默认的输出样式很奇怪哦,是这样子的:
全栈程序员站长
2022/07/22
6740
Java SimpleDateFormat进行日期格式化
Java Date 和 Calendar
Java 语言的Date(日期),Calendar(日历),DateFormat(日期格式)组成了Java标准的一个基本但是非常重要的部分。日期是商业逻辑计算一个关键的部分,所有的开发者都应该能够计算未来的日期,定制日期的显示格式,并将文本数据解析成日期对象。  创建一个日期对象
阳光岛主
2019/02/19
6290
推荐阅读
相关推荐
Java日期处理易踩的十个坑
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验