JavaLocalDate获取星期几数字方法
时间:2026-05-08 20:56:43 235浏览 收藏
本文深入解析了Java中LocalDate获取“周几数字”的实际业务难点——不能简单依赖DayOfWeek枚举的ISO标准值(MONDAY=1至SUNDAY=7),而需根据真实场景(如零售按周日开头、财务按周三起算)动态映射为0~6的业务内偏移量,并借助WeekFields精准计算跨年不误的年周标识(如2024-W22),辅以可复用的分组统计代码和关键避坑指南,帮你彻底规避硬编码、跨年错位、数据库语义不一致等高频陷阱。

Java 的 LocalDate.getDayOfWeek() 返回的是 DayOfWeek 枚举(MONDAY=1 到 SUNDAY=7),但业务中“周次统计”往往不直接对应 ISO 周定义(如周一为每周第一天、周四≥4天才算本周),而是依赖自定义起始日(例如:周日为一周开始,或按财务周从周三起算)。因此不能直接用枚举序号做统计逻辑,需先统一映射到以业务周首日为基准的“本周内偏移量”(0~6)。
明确业务周定义并构建偏移映射表
关键第一步是确定你的业务规则。例如:
- 零售行业常以 周日为每周第一天(即周日=0,周一=1,…,周六=6)
- 某些跨国企业采用 ISO 周(周一为第一天),但要求统计时以“本周四是否在当前自然周内”判定归属(ISO 周判定逻辑)
- 财务系统可能规定 周三为每周起点(即周三=0,周四=1,…,周二=6)
一旦确认起点日(firstDayOfWeek),即可建立 DayOfWeek → 业务周内序号 映射:
// 示例:以周日为每周第 0 天
Map<DayOfWeek, Integer> sundayStartOffset = Map.of(
DayOfWeek.SUNDAY, 0,
DayOfWeek.MONDAY, 1,
DayOfWeek.TUESDAY, 2,
DayOfWeek.WEDNESDAY, 3,
DayOfWeek.THURSDAY, 4,
DayOfWeek.FRIDAY, 5,
DayOfWeek.SATURDAY, 6
);将日期归入业务周,并计算周标识(year-week)
仅靠星期几不够,还需结合日期推算“属于哪一周”。推荐方法:计算该日期所在业务周的**起始日**(LocalDate),再用其年份+第几周生成唯一周标识。
例如,以周日为起点:
- 对任意日期
date,先获取它的DayOfWeek - 计算距最近一个周日的天数偏移:
int daysToSunday = (7 - date.getDayOfWeek().getValue()) % 7(注意:DayOfWeek.getValue() 周日=7,所以需取模调整) - 则本周周日 =
date.minusDays(daysToSunday) - 该周标识可表示为:
weekStart.getYear() + "-W" + String.format("%02d", weekStart.get(WeekFields.SUNDAY_START.weekOfYear()))
更稳健做法是使用 WeekFields(Java 8+):
WeekFields weekFields = WeekFields.of(DayOfWeek.SUNDAY, 1); // 起点周日,最小天数1
TemporalField weekOfYear = weekFields.weekOfYear();
int year = date.get(weekFields.year());
int week = date.get(weekOfYear);
String businessWeekId = String.format("%d-W%02d", year, week);按业务周分组统计的实际代码示例
假设有一组订单日期,需按“周日起点周”统计每周订单数:
List<LocalDate> orderDates = Arrays.asList(
LocalDate.of(2024, 6, 2), // 周日 → 所在周:2024-W22(6.2–6.8)
LocalDate.of(2024, 6, 5), // 周三 → 同属 2024-W22
LocalDate.of(2024, 6, 9) // 下周日 → 2024-W23
);
<p>WeekFields sundayStart = WeekFields.of(DayOfWeek.SUNDAY, 1);</p><p>Map<String, Long> weeklyCount = orderDates.stream()
.collect(Collectors.groupingBy(
d -> String.format("%d-W%02d",
d.get(sundayStart.year()),
d.get(sundayStart.weekOfYear())
),
Collectors.counting()
));</p><p>// 结果:{"2024-W22"=2, "2024-W23"=1}</p>注意事项与避坑点
使用时务必注意以下细节:
- 不要硬编码 1~7 对应周几含义:
DayOfWeek.MONDAY.getValue()==1是 ISO 定义,若业务周从周日开始,MONDAY 应是第 1 天而非第 0 天——需显式转换 - 跨年周容易出错:12月最后一周可能属于下一年的第1周(尤其 ISO 周),用
WeekFields.year()替代LocalDate.getYear() - 避免用“本周一/周日”简单截断:例如
date.with(TemporalAdjusters.previousOrSame(DayOfWeek.MONDAY))在跨月时仍正确,但需确认是否符合业务周边界定义 - 数据库交互时保持语义一致:若 SQL 层也按相同周规则聚合(如 PostgreSQL 的
EXTRACT(WEEK FROM ...)),需确保传参和解析逻辑匹配
到这里,我们也就讲完了《JavaLocalDate获取星期几数字方法》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!
相关阅读
更多>
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
最新阅读
更多>
-
373 收藏
-
235 收藏
-
346 收藏
-
420 收藏
-
324 收藏
-
481 收藏
-
413 收藏
-
233 收藏
-
479 收藏
-
299 收藏
-
148 收藏
-
222 收藏
课程推荐
更多>
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习