大家过年好!


跨年夜除了好好享受假期,系统上从技术上到业务上,都有值得关注的事儿。一年才触发一次,频率低,风险高。更需谨慎。


由于今年是我的本命年。本命年犯太岁。“太岁当头坐 ,无喜必有祸”。本命年顺的话,一顺百顺,鸿运当头,势不可挡。所以今天的标题全用大红色系统跨年的那些事儿_java


经典跨年Bug


系统跨年的那些事儿_微信_02

在2021年的最后两分钟,我怀着忐忑和期待的心情迎接着本命年。并匆忙写下下面的代码:

import cn.hutool.core.date.DateTime;
public void test() throws Exception {
System.out.println(new DateTime().toString("YYYY-MM-dd HH:MM:SS"));
}


运行结果:

2022​-12-31 23:​12:651


很多朋友知道我侧重于稳定性方面的研究,但是最近提这个词提的少了。在​​《架构-稳定性建设逻辑问题实战总结》​​​和​​《稳定性的海因里希法则》​​里我也总结过:稳定性更多的是意识,不是方法。

稳定性方法本质就是两点:




  • 多问几个为什么

  • 多问几个如果发生了某某问题怎么办



这里本质上就是一个“为什么”的问题:所用的技术一定要深入理解。这个运行结果可能和一些人想象的不一致,本质上不是技术上的坑,而是对于日期格式化符号没有理解。

在Java的SimpleDateFormat类注释中有一张表,如下表所示:系统跨年的那些事儿_日期格式化_03

在规定中,y表示year,而Y表示Week Year!


什么是Week Year

我们知道,不同的国家对于一周的开始和结束的定义是不同的。如在中国,我们把星期一作为一周的第一天,而在美国,他们把星期日作为一周的第一天。因为不同人对于日期和时间的表示方法有不同的理解,于是,大家就共同制定了了一个国际规范:ISO 8601 。


国际标准化组织的国际标准ISO 8601是日期和时间的表示方法,全称为《数据存储和交换形式·信息交换·日期和时间的表示方法》。


在 ISO 8601中。week year是说某一周只能属于一个年份。2021年12月31日不能既属于今年的最后一个星期,又是明年的第一个星期。


对于一年的第一个week year有以下四种等效说法:

1,本年度第一个星期四所在的星期

2,1月4日所在的星期

3,本年度第一个至少有4天在同一星期内的星期

4,星期一在去年12月29日至今年1月4日以内的星期


按照这个规范,2021年12月30日是星期四,满足是新一年的第一个星期。那就是说打从2021年12月26日(星期日,ISO 8601中这是一周的第一天)开始,week year就已经是2022年了,新年已经到来了6天了!


日期格式化符号里M大写的代表月份,小写m代表分钟,大写S代表毫秒,小写s代表秒。而一个日期格式化符号出现多少次就代表最少占多少位,不够的前面补0。

public void test() throws Exception {
System.out.println(new DateTime(DateTime.of("2021-12-26","yyyy-MM-dd"))
.toString("yyyy-MM-dd HH:mm:sssss"));
}


运行结果:

2021-12-26 00:00:00000


业务经典问题


2022年1月1日 0点10分。已经收到一些消息:网上反映QQ红包服务器炸了、QQ红包发不出去,拼多多红包提现慢、不能提现,微信红包提示使用人数过多等。说明大家对新年活动的参与度很高嘛。

其实在各个领域都有一些跨年的业务要特殊处理。最常见的一个例子就是“年终决算”。在各个银行,年终决算就是要把账结清。如果一边在做业务,一边在算账,很难算清楚。所以很多系统干脆暂停业务来进行年终决算。


年终决算并不是会计等领域的特定名词。早年的农村生产大队也要进行年终决算。办年终结算首先要列出6个指标来:

1,生产队总收入

2,生产队净收入3,可分配收入4,全年劳动日值,也就是说社员劳动一天折算的工分换成现金有多少钱5,每户社员应当分配的数额

6,每户社员应当分配的实物和现金数量


是不是听起来相当专业,据说那时候每个生产队都会有一两个这方面的高手,精通珠算。让人不禁感叹:高手在民间。