问题背景:
项目之前开发了一个日历功能,有使用到Calendar.WEEK\_OF\_MONTH去获取在当前月的第几周,项目本身有做多语言适配,发现Calendar.WEEK\_OF\_MONTH会跟随系统语言有不一样的效果,这不bug就来了,赶紧排查问题。
由于项目里涉及到的内容过于复杂,赶紧写个小demo试试。
问题排查:
首先贴一下这个月的日历,今天是2023年7月3日。
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")
选取最常用的中文语言和英文语言进行实验,英文分为英国英语和美国英语。
中文语言下结果
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就不会跟随系统语言有不一样的效果了。