下面给我的笔记本电脑上的结果(After)与其他任何地方不同。
private static void prindStartOfWeek()
{
Calendar cal = Calendar.getInstance(TimeZone.getTimeZone("GMT"));
System.out.println(" --------- Before : " + cal.getTime());
cal.set(Calendar.YEAR, 2015);
cal.set(Calendar.WEEK_OF_YEAR, 1);
cal.set(Calendar.DAY_OF_WEEK, Calendar.SUNDAY);
cal.set(Calendar.HOUR, 0);
cal.set(Calendar.MINUTE, 0);
cal.set(Calendar.SECOND, 0);
cal.set(Calendar.MILLISECOND, 0);
System.out.println(" --------- After : " + cal.getTime());
}
我正在使用Eclipse Kepler
,Windows 7 64bit
,Java jdk1.7.0_65
。我也在Eclipse Juno
和cmd (executable jar)
进行了测试。这同样适用。
我已在笔记本电脑上测试了上述代码并打印
--------- Before : Fri Nov 20 14:04:07 EET 2015
--------- After : Sun Jan 04 14:00:00 EET 2015
我已在Microsoft Server 2008
运行Java 7
进行了测试并打印出来(这是正确的值)
--------- Before : Fri Nov 20 14:09:01 EET 2015
--------- After : Sun Dec 28 14:00:00 EET 2014
我已经测试了here too并打印了
--------- Before : Fri Nov 20 10:54:40 UTC 2015
--------- After : Sun Dec 28 00:00:00 UTC 2014
有谁知道可能出现的问题?我的笔记本电脑时区是#34;欧洲\雅典",但我认为这不重要。
此外,这是在另一台计算机上测试的,并打印出正确的值
--------- Before : Fri Nov 20 14:09:01 EET 2015
--------- After : Sun Dec 28 14:00:00 EET 2014
答案 0 :(得分:1)
getCalendar(TimeZone)
"使用指定的时区和默认语言区域获取日历。"
正如@Kartic在上面的评论中指出的那样,在某些地方,星期日被认为是一周的第一天,而在其他地方,星期一是一周的第一天 - 这会受到您的语言环境的影响
您运行代码的计算机之间可能有不同的默认语言环境。
答案 1 :(得分:1)
ZonedDateTime.now() // Capture the current moment as seen through the lens of the wall-clock time used by the people of a particular region (time zone).
.get( IsoFields.WEEK_OF_WEEK_BASED_YEAR ) // Get the week number, with stable results per the ISO 8601 definition of a week that starts on a Monday and considers week # 1 to have the first Thursday of the calendar-year.
在Calendar
类中,“周”的含义按区域设置更改。如果在运行时JVM的当前默认值Locale
不同,则计算周数的结果可能会有所不同。
例如,在美国,一周的第一天是星期天。但在欧洲的大部分地区,以及ISO 8601标准,本周的第一天是星期一。
Calendar
课程很麻烦且令人困惑。幸运的是,现在已经被业界领先的 java.time 类所取代。
以UTC格式捕获当前时刻。
Instant instant = Instant.now() ;
时区对于确定日期至关重要。对于任何给定的时刻,日期在全球范围内因地区而异。例如,在Paris France午夜后的几分钟是新的一天,而Montréal Québec中仍然是“昨天”。
如果未指定时区,则JVM会隐式应用其当前的默认时区。该默认值可能随时更改,因此您的结果可能会有所不同。最好明确指定您期望/预期的时区作为参数。
以continent/region
的格式指定proper time zone name,例如America/Montreal
,Africa/Casablanca
或Pacific/Auckland
。切勿使用诸如EST
或IST
之类的3-4字母缩写,因为它们不是真正的时区,不是标准化的,甚至不是唯一的(!)。
ZoneId z = ZoneId.of( "America/Montreal" ) ;
如果要使用JVM的当前默认时区,请求它并作为参数传递。如果省略,则隐式应用JVM的当前默认值。最好是明确的。
ZoneId z = ZoneId.systemDefault() ; // Get JVM’s current default time zone.
应用ZoneId
来获取ZonedDateTime
对象。
ZonedDateTime zdt = instant.atZone( z ) ;
现在可以查询基于工作周的年份和周。 IsoFields
类提供了对象,通过这些对象我们可以获得符合ISO 8601标准的周数:
实施例
int yearIso = zdt.get( IsoFields.WEEK_BASED_YEAR ) ;
int weekIso = zdt.get( IsoFields.WEEK_OF_WEEK_BASED_YEAR ) ;
提示:考虑将ThreeTen-Extra库添加到您的项目中,以获取对YearWeek
类的访问权限。
java.time框架内置于Java 8及更高版本中。这些类取代了麻烦的旧legacy日期时间类,例如java.util.Date
,Calendar
和& SimpleDateFormat
现在位于Joda-Time的maintenance mode项目建议迁移到java.time类。
要了解详情,请参阅Oracle Tutorial。并搜索Stack Overflow以获取许多示例和解释。规范是JSR 310。
从哪里获取java.time类?
ThreeTen-Extra项目使用其他类扩展java.time。该项目是未来可能添加到java.time的试验场。您可以在此处找到一些有用的课程,例如Interval
,YearWeek
,YearQuarter
和more。