3

我可以轻松计算每月第一天和当前时间之间的时间段:

/**
 * Returns the time range between the first day of month and current time in milliseconds.
 *
 * @param zoneId time zone ID.
 * @return a {@code long} array, where at index: 0 - the first day of month midnight time; 1 - current time.
 */

public static long[] monthDateRange(ZoneId zoneId) {
    long[] toReturn = new long[2];

ZonedDateTime nowZdt = LocalDateTime.now().atZone(zoneId);
ZonedDateTime startZdt = nowZdt.withDayOfMonth(1);
toReturn[0] = startZdt.toInstant().toEpochMilli();
toReturn[1] = nowZdt.toInstant().toEpochMilli();
return toReturn;
}

但是如何在本周的第一天(午夜)开始计数?

4

2 回答 2

4

tl;博士

ZonedDateTime
    .now( ZoneId.of( "Asia/Kolkata" ) )                            // Current moment in a particular time zone.
    .toLocalDate()                                                 // Extract date-only value, losing the time-of-day and time zone components.
    .with( TemporalAdjusters.previousOrSame( DayOfWeek.SUNDAY ) )  // Move to another day-of-week, or same date if this is the desired day-of-week.
    .atStartOfDay( ZoneId.of( "Asia/Kolkata" ) )                   // Determine the first moment of the day. Do *not* assume this time-of-day is 00:00:00 as anomalies such as Daylight Saving Time (DST) may mean otherwise such as 01:00:00. 
    .toInstant()                                                   // Adjust into UTC, same moment, same point on the timeline, but viewed through the lens of UTC time zone.
    .toEpochMilli()                                                // Extract a count-from-epoch in milliseconds. I do *not* recommend tracking date-time this way, but the Question requires this number.

细节

Gruodis的答案很好,但这里有一个更直接和灵活的替代方案。

获取当前时刻作为ZonedDateTime.

ZoneId z = ZoneId.of( "Pacific/Auckland" ) ;
ZonedDateTime now = ZonedDateTime.now( z ) ;

TemporalAdjuster

TemporalAdjuster界面允许您操作日期时间值以获取新的日期时间值。该类TemporalAdjusters(注意复数s)提供了几个方便的实现。使用DayOfWeek枚举来指定您认为哪一天是一周的第一天。

DayOfWeek dowStartOfWeek = DayOfWeek.MONDAY ; 
LocalDate weekStartDate = now.toLocalDate().with( TemporalAdjusters.previousOrSame( DayOfWeek.MONDAY ) ) ;
ZonedDateTime start = weekStartDate.atStartOfDay( z ) ;  // Determine first moment of the day. Note: *not* always 00:00:00.

请参阅在 IdeOne.com 上实时运行的代码

2017-08-21T00:00+12:00[太平洋/奥克兰] 2017-08-21T08:44:46.439+12:00[太平洋/奥克兰]

时间跨度

为了报告您的时间跨度,如果需要, pou 确实可以提取整秒的计数。

long epochSeconds = start.toEpochSecond() ; 

或通过Instant.

long epochMillis = start.toInstant().toEpochMilli() ;

但请记住,这两个数字都会截断任何进一步的小数秒,因为 java.time 类型解析为nanoseconds

除了截断之外,还有其他原因可以避免将 date-time 跟踪为count-from-epoch。由于这些值对人眼来说毫无意义,因此调试要困难得多,并且错误的数据可能不会引起您的注意。此外,您可以假设 epoch 是1970-01-01T00:00:00Z,但至少还有几十个epoch被通用软件系统使用。另一个问题是计数粒度的模糊性,其中一些系统使用整秒,其他使用毫秒,其他使用微秒,其他纳秒,还有一些使用其他分辨率。

Interval

因此long,我建议不要只返回整数,而是返回一个对象。一对Instant对象起作用,这是ThreeTen-Extra项目Interval中的类使用的对象。该类有几个非常方便的方法,我希望调用代码可能会发现有用的方法,例如, , , , ,等等。containsenclosesabutsoverlapsspanisEmpty

org.threeten.extra.Interval interval = Interval.of( start.toInstant() , now.toInstant() ) ;

您可以应用时区来通过一个地区自己的挂钟时间的镜头查看开始或结束。

ZonedDateTime zdtStart = interval.getStart().atZone( z );  // Or `getEnd()`.
于 2017-08-20T20:13:58.573 回答
1

解决方案:

/**
 * Returns the time range between the first day of current week midnight and current time in milliseconds.
 *
 * @param zoneId time zone ID.
 * @return a {@code long} array, where at index: 0 - the first day of current week midnight time; 1 - current time.
 */

public static long[] monthDateRange(ZoneId zoneId) {
    long[] toReturn = new long[2];

//ZonedDateTime nowZdt = LocalDateTime.now().atZone(zoneId);
ZonedDateTime nowZdt = ZonedDateTime.now(zoneId);//As suggested by Basil Bourque (tested).
//ZonedDateTime startZdt = nowZdt.with(ChronoField.DAY_OF_WEEK, 1);
ZonedDateTime startZdt = nowZdt.with(TemporalAdjusters.previousOrSame(DayOfWeek.MONDAY));//As suggested by Basil Bourque (tested).
startZdt = startZdt.toLocalDate ().atStartOfDay(zoneId);
toReturn[0] = startZdt.toInstant().toEpochMilli();
toReturn[1] = nowZdt.toInstant().toEpochMilli();
return toReturn;
}

请参阅在 IdeOne.com 上实时运行的代码

于 2017-08-20T18:35:31.107 回答