问题背景:


项目之前开发了一个日历功能,有使用到Calendar.WEEK\_OF\_MONTH去获取在当前月的第几周,项目本身有做多语言适配,发现Calendar.WEEK\_OF\_MONTH会跟随系统语言有不一样的效果,这不bug就来了,赶紧排查问题。

由于项目里涉及到的内容过于复杂,赶紧写个小demo试试。

问题排查:


首先贴一下这个月的日历,今天是2023年7月3日。

Calendar.WEEK_OF_MONTH跟着系统语言变? _问题排查

7月1日为周六 7月2日为周日 7月3日为周一 7月共涉及到6个周

示例代码

//2023.07.01,周六
val c1 = Calendar.getInstance().apply {
    timeInMillis = 1688140800000
}
val totalWeek = c1.getActualMaximum(Calendar.WEEK_OF_MONTH)
Log.d("week", "totalWeek $totalWeek")
val week1 = c1.get(Calendar.WEEK_OF_MONTH)
Log.d("week", "week1: $week1")
//2023.07.02,周日
c1.add(Calendar.DATE, 1)
val week2 = c1.get(Calendar.WEEK_OF_MONTH)
Log.d("week", "week2: $week2")
//2023.07.03,周一
c1.add(Calendar.DATE, 1)
val week3 = c1.get(Calendar.WEEK_OF_MONTH)
Log.d("week", "week3: $week3")

Calendar.WEEK_OF_MONTH跟着系统语言变? _问题排查_02

选取最常用的中文语言和英文语言进行实验,英文分为英国英语和美国英语。

中文语言下结果

totalWeek: 6
week1: 1
week2: 2
week3: 2

共有6周,1号周六为第一周,2号周日为第二周,3号周一为第二周,这就是我们常规理解的每周从周日开始,周日为一周的第一天。

美国英语语言下结果

totalWeek: 6
week1: 1
week2: 2
week3: 2

同中文语言下结果

英国英语语言下结果

totalWeek: 5
week1: 0
week2: 0
week3: 1

共有5周,竟然少了一周,1号周六为第0周,2号周日为第0周,3号周一为第一周,从3号周一开始才算是该月的第一周。

问题原因

从上述结果大概可以猜测出来,应该有两方面的标准不一样,一是每周从周几开始,周几才算是一周的第一天,二是怎样才算是第一周开始了。

查阅了大量资料之后,在这个问题中找到了答案: stackoverflow.com/questions/6…

英国采用的标准是每周从周一开始,且每月的第一周是该月至少包含4天的第一个周。

问题解决

知道了问题原因之后,一对一地解决即可:

c1.firstDayOfWeek = Calendar.SUNDAY
c1.minimalDaysInFirstWeek = 1

通过firstDayOfWeek方法设置Calendar的每周从周日开始,minimalDaysInFirstWeek方法设置Calendar的第一周最少天数为1。

设置之后,Calendar.WEEK\_OF\_MONTH就不会跟随系统语言有不一样的效果了。