在看王晓华所著《算法的乐趣》(王晓华著. 算法的乐趣[M]. 北京:人民邮电出版社, 2015.04.)第11章中给出的农历程序源代码时,发如今农历置闰方面彷佛有点问题,值得商榷。算法
书中给出的农历闰月设置规则与《GB/T 33661-2017 农历的编算和颁行》相一致,但长篇大论的比国标长多了,因此直接引用国标原文吧:数组
4 农历的编排规则
4.1 以北京时间为标准时间。
4.2 朔日为农历月的第一个农历日。
4.3 包含节气冬至在内的农历月为农历十一月。
4.4 若从某个农历十一月开始到下一个农历十一月(不含)之间有13个农历月,则须要置闰。置闰规则为:取其中最早出现的一个不包含中气的农历月为农历闰月。
4.5 农历十一月以后第2个(不计闰月)农历月为农历年的起始月。函数
书中判断是否须要置闰的代码是CChineseCalendar::CalcLeapChnMonth(),没有该书或没有该书源代码的能够看这里,有相同的文字说明和源代码:
https://blog.csdn.net/orbit/article/details/9337377.net
CalcLeapChnMonth函数代码初看没有问题,与国标中4.4相一致,但其实存在一个逻辑漏洞: 代码中的m_NewMoonJD数组是从前一年的十一月(冬至月)开始的,所以代码blog
if(int(m_NewMoonJD[13] + 0.5) <= int(m_SolarTermsJD[24] + 0.5)) //第13月的月末没有超过冬至,说明今年须要闰一个月it
考虑了本年前11个月是否须要置闰,但漏掉了另一个考虑选项:本年的第12个月(m_NewMoonJD[14])是否须要置闰?引用
要判断本年第12个月是否须要置闰(闰冬月),就须要从本年冬至月计算到下年冬至月,发现有13个月时再看本年第12个月是否无中气。简单一点说,其实就是在用书中的代码计算发现本年无闰月时,还要用相同的代码再算一遍第二年的闰月看是否是本年的第十二个月。第二遍计算时m_NewMoonJD数组中存储的是从本年冬至到下年冬至之间的朔时刻。而在用现有的代码计算发现本年须要置闰时,也要检查一下去年的冬月是否已经闰过,若是闰过就别再闰了。程序
若是不作这样的修正,在碰到2033年这样须要闰冬月的年份就会出现错误。时间