Android 7.x (API24) WEEK_OF_MONTH 日历错误?
Android 7.x (API24) WEEK_OF_MONTH Calendar Bug?
我们在 Android 7 (API 24/25) 使用 Calendar
.
时出现了一些奇怪的行为
给出这个相当简单的代码:
SimpleDateFormat month_date = new SimpleDateFormat("dd.MM.YYYY");
Calendar cal = new GregorianCalendar(TimeZone.getTimeZone("Europe/Berlin"), Locale.GERMANY);
cal.setFirstDayOfWeek(Calendar.MONDAY);
for (int month = Calendar.JANUARY; month <= Calendar.DECEMBER; month++) {
Calendar start = ((Calendar) cal.clone());
start.set(2017, month, 1);
Calendar end = ((Calendar) start.clone());
end.set(Calendar.DAY_OF_MONTH, end.getActualMaximum(Calendar.DAY_OF_MONTH));
Log.d("CAL", "\n Date Start: " + month_date.format(start.getTime()) + " " +
" WEEK_OF_MONTH: " + start.get(Calendar.WEEK_OF_MONTH)
);
Log.d("CAL", "\n Date End: " + month_date.format(end.getTime()) +
" WEEK_OF_MONTH: " + end.get(Calendar.WEEK_OF_MONTH)
);
}
运行 Android 4.0x,5.x 和 6.x 显示 WEEK_OF_MONTH
:
的正确值
Date Start: 01.01.2016 WEEK_OF_MONTH: 1
Date End: 31.01.2017 WEEK_OF_MONTH: 6
Date Start: 01.02.2017 WEEK_OF_MONTH: 1
Date End: 28.02.2017 WEEK_OF_MONTH: 5
Date Start: 01.03.2017 WEEK_OF_MONTH: 1
Date End: 31.03.2017 WEEK_OF_MONTH: 5
Date Start: 01.04.2017 WEEK_OF_MONTH: 1
Date End: 30.04.2017 WEEK_OF_MONTH: 5
Date Start: 01.05.2017 WEEK_OF_MONTH: 1
Date End: 31.05.2017 WEEK_OF_MONTH: 5
Date Start: 01.06.2017 WEEK_OF_MONTH: 1
Date End: 30.06.2017 WEEK_OF_MONTH: 5
Date Start: 01.07.2017 WEEK_OF_MONTH: 1
Date End: 31.07.2017 WEEK_OF_MONTH: 6
Date Start: 01.08.2017 WEEK_OF_MONTH: 1
Date End: 31.08.2017 WEEK_OF_MONTH: 5
Date Start: 01.09.2017 WEEK_OF_MONTH: 1
Date End: 30.09.2017 WEEK_OF_MONTH: 5
Date Start: 01.10.2017 WEEK_OF_MONTH: 1
Date End: 31.10.2017 WEEK_OF_MONTH: 6
Date Start: 01.11.2017 WEEK_OF_MONTH: 1
Date End: 30.11.2017 WEEK_OF_MONTH: 5
Date Start: 01.12.2017 WEEK_OF_MONTH: 1
Date End: 31.12.2017 WEEK_OF_MONTH: 5
运行 上 Android 7.x 损坏 WEEK_OF_MONTH
:
Date Start: 01.01.2016 WEEK_OF_MONTH: 0
Date End: 31.01.2017 WEEK_OF_MONTH: 5
Date Start: 01.02.2017 WEEK_OF_MONTH: 1
Date End: 28.02.2017 WEEK_OF_MONTH: 5
Date Start: 01.03.2017 WEEK_OF_MONTH: 1
Date End: 31.03.2017 WEEK_OF_MONTH: 5
Date Start: 01.04.2017 WEEK_OF_MONTH: 0
Date End: 30.04.2017 WEEK_OF_MONTH: 4
Date Start: 01.05.2017 WEEK_OF_MONTH: 1
Date End: 31.05.2017 WEEK_OF_MONTH: 5
Date Start: 01.06.2017 WEEK_OF_MONTH: 1
Date End: 30.06.2017 WEEK_OF_MONTH: 5
Date Start: 01.07.2017 WEEK_OF_MONTH: 0
Date End: 31.07.2017 WEEK_OF_MONTH: 5
Date Start: 01.08.2017 WEEK_OF_MONTH: 1
Date End: 31.08.2017 WEEK_OF_MONTH: 5
Date Start: 01.09.2017 WEEK_OF_MONTH: 0
Date End: 30.09.2017 WEEK_OF_MONTH: 4
Date Start: 01.10.2017 WEEK_OF_MONTH: 0
Date End: 31.10.2017 WEEK_OF_MONTH: 5
Date Start: 01.11.2017 WEEK_OF_MONTH: 1
Date End: 30.11.2017 WEEK_OF_MONTH: 5
Date Start: 01.12.2017 WEEK_OF_MONTH: 0
Date End: 31.12.2017 WEEK_OF_MONTH: 4
我们在 API 24.
中找不到 Calendar
的任何记录更改
知道如何解决这个问题吗?
我们已经为 WEEK_OF_MONTH
实现了自己的方法,返回与 Android 6.x 相同的值,但看起来 set(WEEK_OF_MONTH)
也被破坏了。
使用 Joda 重写代码不是一种选择。
您可以使用setMinimalDaysInFirstWeek()
方法change how the weeks are counted:
Calendar cal = new GregorianCalendar(TimeZone.getTimeZone("Europe/Berlin"), Locale.GERMANY);
cal.setFirstDayOfWeek(Calendar.MONDAY);
cal.setMinimalDaysInFirstWeek(1);
这将为月周生成正确的值。
你说过用 Joda-Time 重写代码不是一种选择,但是另一个 API 呢?
在Android中你可以使用ThreeTen Backport, a great backport for Java 8's new date/time classes, together with the ThreeTenABP (more on how to use it ).
顺便说一句,Joda-Time 处于维护模式,正在被新的 APIs 取代,所以我不建议用它开始一个新项目.即使在 joda's website 它说:"Note that Joda-Time is considered to be a largely “finished” project. No major enhancements are planned. If using Java SE 8, please migrate to java.time (JSR-310).".
下面的类在org.threeten.bp
包下。代码将是这样的:
import java.util.Locale;
import org.threeten.bp.DayOfWeek;
import org.threeten.bp.LocalDate;
import org.threeten.bp.Month;
import org.threeten.bp.format.DateTimeFormatter;
import org.threeten.bp.format.DateTimeFormatterBuilder;
import org.threeten.bp.temporal.TemporalAdjusters;
import org.threeten.bp.temporal.WeekFields;
DateTimeFormatter fmt = new DateTimeFormatterBuilder()
// day and month
.appendPattern("dd.MM.")
// week based year (equivalent to YYYY in SimpleDateFormat)
.appendValue(WeekFields.ISO.weekBasedYear())
// create formatter
.toFormatter(Locale.GERMANY);
// week starting at monday, consider week=1 even if it has 1 day (default is 4)
WeekFields wf = WeekFields.of(DayOfWeek.MONDAY, 1);
for (Month month : Month.values()) {
LocalDate start = LocalDate.of(2017, month, 1);
LocalDate end = start.with(TemporalAdjusters.lastDayOfMonth());
System.out.println("Date Start: " + fmt.format(start) + " " + " WEEK_OF_MONTH: " + start.get(wf.weekOfMonth()));
System.out.println("Date End: " + fmt.format(end) + " WEEK_OF_MONTH: " + end.get(wf.weekOfMonth()));
}
输出将是:
Date Start: 01.01.2016 WEEK_OF_MONTH: 1
Date End: 31.01.2017 WEEK_OF_MONTH: 6
Date Start: 01.02.2017 WEEK_OF_MONTH: 1
Date End: 28.02.2017 WEEK_OF_MONTH: 5
Date Start: 01.03.2017 WEEK_OF_MONTH: 1
Date End: 31.03.2017 WEEK_OF_MONTH: 5
Date Start: 01.04.2017 WEEK_OF_MONTH: 1
Date End: 30.04.2017 WEEK_OF_MONTH: 5
Date Start: 01.05.2017 WEEK_OF_MONTH: 1
Date End: 31.05.2017 WEEK_OF_MONTH: 5
Date Start: 01.06.2017 WEEK_OF_MONTH: 1
Date End: 30.06.2017 WEEK_OF_MONTH: 5
Date Start: 01.07.2017 WEEK_OF_MONTH: 1
Date End: 31.07.2017 WEEK_OF_MONTH: 6
Date Start: 01.08.2017 WEEK_OF_MONTH: 1
Date End: 31.08.2017 WEEK_OF_MONTH: 5
Date Start: 01.09.2017 WEEK_OF_MONTH: 1
Date End: 30.09.2017 WEEK_OF_MONTH: 5
Date Start: 01.10.2017 WEEK_OF_MONTH: 1
Date End: 31.10.2017 WEEK_OF_MONTH: 6
Date Start: 01.11.2017 WEEK_OF_MONTH: 1
Date End: 30.11.2017 WEEK_OF_MONTH: 5
Date Start: 01.12.2017 WEEK_OF_MONTH: 1
Date End: 31.12.2017 WEEK_OF_MONTH: 5
我们在 Android 7 (API 24/25) 使用 Calendar
.
给出这个相当简单的代码:
SimpleDateFormat month_date = new SimpleDateFormat("dd.MM.YYYY");
Calendar cal = new GregorianCalendar(TimeZone.getTimeZone("Europe/Berlin"), Locale.GERMANY);
cal.setFirstDayOfWeek(Calendar.MONDAY);
for (int month = Calendar.JANUARY; month <= Calendar.DECEMBER; month++) {
Calendar start = ((Calendar) cal.clone());
start.set(2017, month, 1);
Calendar end = ((Calendar) start.clone());
end.set(Calendar.DAY_OF_MONTH, end.getActualMaximum(Calendar.DAY_OF_MONTH));
Log.d("CAL", "\n Date Start: " + month_date.format(start.getTime()) + " " +
" WEEK_OF_MONTH: " + start.get(Calendar.WEEK_OF_MONTH)
);
Log.d("CAL", "\n Date End: " + month_date.format(end.getTime()) +
" WEEK_OF_MONTH: " + end.get(Calendar.WEEK_OF_MONTH)
);
}
运行 Android 4.0x,5.x 和 6.x 显示 WEEK_OF_MONTH
:
Date Start: 01.01.2016 WEEK_OF_MONTH: 1
Date End: 31.01.2017 WEEK_OF_MONTH: 6
Date Start: 01.02.2017 WEEK_OF_MONTH: 1
Date End: 28.02.2017 WEEK_OF_MONTH: 5
Date Start: 01.03.2017 WEEK_OF_MONTH: 1
Date End: 31.03.2017 WEEK_OF_MONTH: 5
Date Start: 01.04.2017 WEEK_OF_MONTH: 1
Date End: 30.04.2017 WEEK_OF_MONTH: 5
Date Start: 01.05.2017 WEEK_OF_MONTH: 1
Date End: 31.05.2017 WEEK_OF_MONTH: 5
Date Start: 01.06.2017 WEEK_OF_MONTH: 1
Date End: 30.06.2017 WEEK_OF_MONTH: 5
Date Start: 01.07.2017 WEEK_OF_MONTH: 1
Date End: 31.07.2017 WEEK_OF_MONTH: 6
Date Start: 01.08.2017 WEEK_OF_MONTH: 1
Date End: 31.08.2017 WEEK_OF_MONTH: 5
Date Start: 01.09.2017 WEEK_OF_MONTH: 1
Date End: 30.09.2017 WEEK_OF_MONTH: 5
Date Start: 01.10.2017 WEEK_OF_MONTH: 1
Date End: 31.10.2017 WEEK_OF_MONTH: 6
Date Start: 01.11.2017 WEEK_OF_MONTH: 1
Date End: 30.11.2017 WEEK_OF_MONTH: 5
Date Start: 01.12.2017 WEEK_OF_MONTH: 1
Date End: 31.12.2017 WEEK_OF_MONTH: 5
运行 上 Android 7.x 损坏 WEEK_OF_MONTH
:
Date Start: 01.01.2016 WEEK_OF_MONTH: 0
Date End: 31.01.2017 WEEK_OF_MONTH: 5
Date Start: 01.02.2017 WEEK_OF_MONTH: 1
Date End: 28.02.2017 WEEK_OF_MONTH: 5
Date Start: 01.03.2017 WEEK_OF_MONTH: 1
Date End: 31.03.2017 WEEK_OF_MONTH: 5
Date Start: 01.04.2017 WEEK_OF_MONTH: 0
Date End: 30.04.2017 WEEK_OF_MONTH: 4
Date Start: 01.05.2017 WEEK_OF_MONTH: 1
Date End: 31.05.2017 WEEK_OF_MONTH: 5
Date Start: 01.06.2017 WEEK_OF_MONTH: 1
Date End: 30.06.2017 WEEK_OF_MONTH: 5
Date Start: 01.07.2017 WEEK_OF_MONTH: 0
Date End: 31.07.2017 WEEK_OF_MONTH: 5
Date Start: 01.08.2017 WEEK_OF_MONTH: 1
Date End: 31.08.2017 WEEK_OF_MONTH: 5
Date Start: 01.09.2017 WEEK_OF_MONTH: 0
Date End: 30.09.2017 WEEK_OF_MONTH: 4
Date Start: 01.10.2017 WEEK_OF_MONTH: 0
Date End: 31.10.2017 WEEK_OF_MONTH: 5
Date Start: 01.11.2017 WEEK_OF_MONTH: 1
Date End: 30.11.2017 WEEK_OF_MONTH: 5
Date Start: 01.12.2017 WEEK_OF_MONTH: 0
Date End: 31.12.2017 WEEK_OF_MONTH: 4
我们在 API 24.
中找不到Calendar
的任何记录更改
知道如何解决这个问题吗?
我们已经为 WEEK_OF_MONTH
实现了自己的方法,返回与 Android 6.x 相同的值,但看起来 set(WEEK_OF_MONTH)
也被破坏了。
使用 Joda 重写代码不是一种选择。
您可以使用setMinimalDaysInFirstWeek()
方法change how the weeks are counted:
Calendar cal = new GregorianCalendar(TimeZone.getTimeZone("Europe/Berlin"), Locale.GERMANY);
cal.setFirstDayOfWeek(Calendar.MONDAY);
cal.setMinimalDaysInFirstWeek(1);
这将为月周生成正确的值。
你说过用 Joda-Time 重写代码不是一种选择,但是另一个 API 呢?
在Android中你可以使用ThreeTen Backport, a great backport for Java 8's new date/time classes, together with the ThreeTenABP (more on how to use it
顺便说一句,Joda-Time 处于维护模式,正在被新的 APIs 取代,所以我不建议用它开始一个新项目.即使在 joda's website 它说:"Note that Joda-Time is considered to be a largely “finished” project. No major enhancements are planned. If using Java SE 8, please migrate to java.time (JSR-310).".
下面的类在org.threeten.bp
包下。代码将是这样的:
import java.util.Locale;
import org.threeten.bp.DayOfWeek;
import org.threeten.bp.LocalDate;
import org.threeten.bp.Month;
import org.threeten.bp.format.DateTimeFormatter;
import org.threeten.bp.format.DateTimeFormatterBuilder;
import org.threeten.bp.temporal.TemporalAdjusters;
import org.threeten.bp.temporal.WeekFields;
DateTimeFormatter fmt = new DateTimeFormatterBuilder()
// day and month
.appendPattern("dd.MM.")
// week based year (equivalent to YYYY in SimpleDateFormat)
.appendValue(WeekFields.ISO.weekBasedYear())
// create formatter
.toFormatter(Locale.GERMANY);
// week starting at monday, consider week=1 even if it has 1 day (default is 4)
WeekFields wf = WeekFields.of(DayOfWeek.MONDAY, 1);
for (Month month : Month.values()) {
LocalDate start = LocalDate.of(2017, month, 1);
LocalDate end = start.with(TemporalAdjusters.lastDayOfMonth());
System.out.println("Date Start: " + fmt.format(start) + " " + " WEEK_OF_MONTH: " + start.get(wf.weekOfMonth()));
System.out.println("Date End: " + fmt.format(end) + " WEEK_OF_MONTH: " + end.get(wf.weekOfMonth()));
}
输出将是:
Date Start: 01.01.2016 WEEK_OF_MONTH: 1
Date End: 31.01.2017 WEEK_OF_MONTH: 6
Date Start: 01.02.2017 WEEK_OF_MONTH: 1
Date End: 28.02.2017 WEEK_OF_MONTH: 5
Date Start: 01.03.2017 WEEK_OF_MONTH: 1
Date End: 31.03.2017 WEEK_OF_MONTH: 5
Date Start: 01.04.2017 WEEK_OF_MONTH: 1
Date End: 30.04.2017 WEEK_OF_MONTH: 5
Date Start: 01.05.2017 WEEK_OF_MONTH: 1
Date End: 31.05.2017 WEEK_OF_MONTH: 5
Date Start: 01.06.2017 WEEK_OF_MONTH: 1
Date End: 30.06.2017 WEEK_OF_MONTH: 5
Date Start: 01.07.2017 WEEK_OF_MONTH: 1
Date End: 31.07.2017 WEEK_OF_MONTH: 6
Date Start: 01.08.2017 WEEK_OF_MONTH: 1
Date End: 31.08.2017 WEEK_OF_MONTH: 5
Date Start: 01.09.2017 WEEK_OF_MONTH: 1
Date End: 30.09.2017 WEEK_OF_MONTH: 5
Date Start: 01.10.2017 WEEK_OF_MONTH: 1
Date End: 31.10.2017 WEEK_OF_MONTH: 6
Date Start: 01.11.2017 WEEK_OF_MONTH: 1
Date End: 30.11.2017 WEEK_OF_MONTH: 5
Date Start: 01.12.2017 WEEK_OF_MONTH: 1
Date End: 31.12.2017 WEEK_OF_MONTH: 5