如何将一年中的几周转换为 LocalDate

How to convert weeks of year to LocalDate

我有一个包含年-周的字符串,例如“2015-40”和年月格式,例如“2015-08”,想在 Scala 中转换为 LocalDate。

我试过使用

val date = "2015-40"
val formatter = DateTimeFormatter.ofPattern("yyyy-ww") 
LocalDate.parse(date, formatter)

但最终出现 DateTimeParseException 错误。非常感谢有关如何执行此操作的任何帮助

提前致谢

LocalDate 有年、月、日三部分。因此,对于 year-month 字符串,您必须根据您的要求在特定日期获取 LocalDate。如演示代码所示,解析 year-month 字符串非常简单。

对于 year-week 字符串,您必须在一周中的特定日期获取 LocalDate,例如星期一或今天等。此外,我发现获取年份和星期然后使用方法 LocalDate 获取所需的 LocalDate.[=19 实例而不是直接解析字符串更容易=]

import java.time.DayOfWeek;
import java.time.LocalDate;
import java.time.YearMonth;
import java.time.format.DateTimeFormatter;
import java.time.temporal.TemporalAdjusters;
import java.time.temporal.WeekFields;

public class Main {
    public static void main(String[] args) {
        //#################### Year-Month #######################
        // Given year-month string
        var yearMonthStr = "2015-08";

        // LocalDate parsed from yearMonthStr and on the 1st day of the month
        LocalDate date2 = YearMonth.parse(yearMonthStr, DateTimeFormatter.ofPattern("u-M")).atDay(1);
        System.out.println(date2);

        // LocalDate parsed from yearMonthStr and on the last day of the month
        date2 = YearMonth.parse(yearMonthStr, DateTimeFormatter.ofPattern("u-M")).atEndOfMonth();
        System.out.println(date2);

        // LocalDate parsed from yearMonthStr and on specific day of the month
        date2 = YearMonth.parse(yearMonthStr, DateTimeFormatter.ofPattern("u-M")).atDay(1).withDayOfMonth(10);
        System.out.println(date2);

        
        //#################### Year-Week #######################
        // Given year-week string
        var yearWeekStr = "2015-40";

        // Split the string on '-' and get year and week values
        String[] parts = yearWeekStr.split("-");
        int year = Integer.parseInt(parts[0]);
        int week = Integer.parseInt(parts[1]);

        // LocalDate with year, week and today's day e.g. Fri
        LocalDate date1 = LocalDate.now()
                            .withYear(year)
                            .with(WeekFields.ISO.weekOfYear(), week);
        System.out.println(date1);

        // LocalDate with year, week and next Mon (or same if today is Mon)
        date1 = LocalDate.now()
                .withYear(year)
                .with(WeekFields.ISO.weekOfYear(), week)
                .with(TemporalAdjusters.nextOrSame(DayOfWeek.MONDAY));
        System.out.println(date1);

        // LocalDate with year, week and today's day previous Mon (or same if today is Mon)
        date1 = LocalDate.now()
                .withYear(year)
                .with(WeekFields.ISO.weekOfYear(), week)
                .with(TemporalAdjusters.previousOrSame(DayOfWeek.MONDAY));
        System.out.println(date1);
    }
}

输出:

2015-08-01
2015-08-31
2015-08-10
2015-10-02
2015-10-05
2015-09-28
    String date = "2015-40";
    DateTimeFormatter formatter = new DateTimeFormatterBuilder()
            .appendPattern("YYYY-ww")
            .parseDefaulting(ChronoField.DAY_OF_WEEK, DayOfWeek.MONDAY.getValue())
            .toFormatter(Locale.FRANCE);
    LocalDate ld = LocalDate.parse(date, formatter);
    
    System.out.println(ld);

对不起,我只能写Java,我相信你能翻译成Scala。输出为:

2015-09-28

你为什么得到异常?我们缺少一些东西来将 2015-40 解析为 LocalDate:

  • A LocalDate 是日历日期,第 40 周由 7 天组成。 Java不知道你想要那7天中的哪一天,拒绝为你做出选择。在我上面的代码中,我指定了星期一。一周中的其他任何一天都可以。
  • 有点微妙。虽然对于人类来说,2015 年和第 40 周是明确的,但新年前后情况并非总是如此,第 1 周可能在新年之前开始,或者第 52 或 53 周可能在新年之后延长。因此,日历年和周数并不总是定义一个特定的周。相反,我们需要 周年week-based 年 的概念。一年从第 1 周开始持续一周,无论这是否意味着它在新年之前或之后几天开始。它一直持续到上周的最后一天,最常见的是在新年前或新年后的几天。要告诉 DateTimeFormatter 我们要解析(或打印)周年,我们需要使用大写字母 YYYY 而不是小写字母 yyyy(或 uuuu)。

顺便说一句,如果您可以影响格式,请考虑 2015-W40W。这是年和周的 ISO 8601 格式。在 ISO 中,2015-12 表示年份和月份,许多人会这样理解。所以为了消除歧义,避免误读。

编辑:

在我的解释中,我采用了 ISO 周计划(星期一是一周的第一天,第一周定义为新年中至少有 4 天的第一周)。您可以将不同的区域设置传递给格式化程序构建器以获得不同的周计划。

如果您确定您的周遵循 ISO,Andreas 在评论中的建议就足够了,我们希望作为答案的一部分:

Alternatively, add the ThreeTen Extra library, so you can use the YearWeek class, that has a nice atDay​(DayOfWeek dayOfWeek) method for getting a LocalDate.

Link: Wikipedia article: ISO 8601