1

我需要找到接下来的 n 个星期五,但想排除第 4 个和可能的第 5 个星期五,即我想得到接下来的 n 个星期五,它们是每个月的第 1 个、第 2 个或第 3 个星期五。我找到了一种在接下来的 n 个星期五(例如接下来的 10 个星期五)进行流式传输的方法:

LocalDate now = LocalDate.now();
Stream.iterate(now , localDate -> localDate.with(TemporalAdjusters.next(DayOfWeek.FRIDAY)))
     .skip(1) // skip today 
     .limit(10)
     .forEach(System.out::println);

输出:

2021-10-22
2021-10-29
2021-11-05
2021-11-12
2021-11-19
2021-11-26
2021-12-03
2021-12-10
2021-12-17
2021-12-24

但我需要排除例如2021-10-22&2021-10-29因为它们是 10 月的第 4 和第 5 个星期五。同样2021-11-26&2021-12-24因为它们不是第一个、第二个或第三个星期五。

我搜索了 SO 并找到了一些方法来计算一个月的第 n 个星期五,例如这个

public static LocalDate nthFridayOfMonth(LocalDate localDate, int week) {
    return localDate.with(TemporalAdjusters.dayOfWeekInMonth(week, DayOfWeek.FRIDAY));
}

返回给定日期的第 n 个星期五。

如果给定的 LocalDate 是一个月的第一个、第二个或第三个星期五,如何修改此方法以使其返回布尔值?或者使用这个方法来实现另一个我需要的方法?

理想情况下,我想使用上面的流并使用一种方法,该方法返回一个布尔值作为过滤器:

LocalDate now = LocalDate.now();
Stream.iterate(now , localDate -> localDate.with(TemporalAdjusters.next(DayOfWeek.FRIDAY)))
     .skip(1) // skip today 
     .filter(d -> is1st2ndOr3rdFriday(d))   
     .limit(10)
     .forEach(System.out::println);

要得到

2021-11-05
2021-11-12
2021-11-19
2021-12-03
2021-12-10
2021-12-17
2022-01-07
2022-01-14
2022-01-21
2022-02-04

感谢有关如何实现过滤器方法的任何帮助boolean is1st2ndOr3rdFriday(LocalDate d) {}

4

3 回答 3

6

可以获取ALIGNED_WEEK_OF_MONTH本地日期的时域,判断是否≤3。

Stream.iterate(now , localDate -> localDate.with(TemporalAdjusters.next(DayOfWeek.FRIDAY)))
    .skip(1) // skip today
    .filter(x -> x.get(ChronoField.ALIGNED_WEEK_OF_MONTH) <= 3)
    .limit(10)
    .forEach(System.out::println);
于 2021-10-17T15:01:41.533 回答
2

另一种选择:迭代几个月

为了变化,这里有一个不同的想法。我正在创建一个几个月的流,并明确地从每个月的前 3 个星期五开始。

    LocalDate today = LocalDate.now(ZoneId.systemDefault());
    YearMonth thisMonth = YearMonth.from(today);
    List<LocalDate> fridays = Stream.iterate(thisMonth, ym -> ym.plusMonths(1))
            .flatMap(ym -> Stream.of(fridayOfMonth(ym, 1), fridayOfMonth(ym, 2), fridayOfMonth(ym, 3)))
            .filter(date -> date.isAfter(today))
            .limit(10)
            .collect(Collectors.toList());
    System.out.println(fridays);

今天运行时的输出:

[2021-11-05、2021-11-12、2021-11-19、2021-12-03、2021-12-10、2021-12-17、2022-01-07、2022-01-14、2022 -01-21, 2022-02-04]

我使用以下辅助方法来查找一个月的第 n 个星期五:

private static LocalDate fridayOfMonth(YearMonth ym, int no) {
    return ym.atDay(15).with(DayOfWeek.FRIDAY).with(ChronoField.ALIGNED_WEEK_OF_MONTH, no);
}

一个变化:仍然为了变化,这里有不同的方法来创建一个月的前三个星期五的流:

    List<LocalDate> fridays = Stream.iterate(thisMonth, ym -> ym.plusMonths(1))
            .flatMap(MyClass::firstThreeFridays)
            .filter(date -> date.isAfter(today))
            .limit(10)
            .collect(Collectors.toList());

这次我的辅助方法是这样的:

private static Stream<LocalDate> firstThreeFridays(YearMonth ym) {
    LocalDate firstFriday = ym.atDay(1).with(TemporalAdjusters.nextOrSame(DayOfWeek.FRIDAY));
    return Stream.of(firstFriday, firstFriday.plusWeeks(1), firstFriday.plusWeeks(2));
}

结果和以前一样。

(迭代几个月的想法也在 Basil Bourque 的一个现已删除的答案中。)

于 2021-10-26T19:46:32.067 回答
1

另一种解决方案:您可以按月分组您的日期(到 Map<Month, List<LocalDate >> ),然后对于每个月的键,如果有超过 3 个日期,将其删除,并返回结果:

public static void main(String[] args) {
    // initialize your date here:
    LocalDate now = LocalDate.now();
    // Grouping Dates By Month 
    Map<Month, List<LocalDate>> map = Stream
            .iterate(LocalDate.of(now.getYear(), now.getMonth(), 1),
                    localDate -> localDate.with(TemporalAdjusters.next(DayOfWeek.FRIDAY)))
            .limit(20).collect(Collectors.groupingBy(element -> element.getMonth()));
    // delete the 4 or 5 fridays
    map.forEach((k, v) -> {
        if (v.size() > 3)
            v.removeIf(e -> e.isBefore(now) || v.indexOf(e) > 2);
    });
    // print results sorted
    Stream.of(map.values()).flatMap(e -> e.stream()).flatMap(List::stream).sorted().limit(10)
            .forEach(System.out::println);
}

输出

2021-11-05

2021-11-12

2021-11-19

2021-12-03

2021-12-10

2021-12-17

2022-01-07

2022-01-14

2022-01-21

2022-02-04

于 2021-10-17T15:20:56.180 回答