我对Java很新,我正在尝试构建一个以下面的格式打印日历的程序。它应该使用当前日期启动日历,并且仅提前1个月创建日历。
Example: October 26, 2016. Su M T W T F S [October] | | | | | 27 | 28 | 29 | | 30 | 31 | | | | | | [November] | | | 1 | 2 | 3 | 4 | 5 | | 6 | 7 | 8 | 9 | 10 | 11 | 12 | | 13 | 14 | 15 | 16 | 17 | 18 | 19 | | 20 | 21 | 22 | 23 | 24 | 25 | 26 |
我尽可能多地写了,但在进行实际编码时我一直遇到问题。例如,我无法弄清楚如何在一周中的正确日期启动日历,而我在一个月内打印正确的周数时遇到问题。我应该重做我的代码,以便我使用calendar = new GregorianCalendar()吗?或者我应该只使用instanceOf()并继续搞乱Calendar类?
这是我可以编写的代码,而不会破坏我的日历。
StringBuilder calendarString = new StringBuilder();
Calendar calendar = Calendar.getInstance();
int maxDaysOfMonth = calendar.getActualMaximum(Calendar.DAY_OF_MONTH);
int dayOfMonth = Calendar.DAY_OF_WEEK;
int weekOfMonth = Calendar.WEEK_OF_MONTH;
int theMonth = Calendar.MONTH;
String month;
switch(theMonth) { // Months start with 0
case 0: month = "January"; break;
case 1: month = "February"; break;
case 2: month = "March"; break;
case 3: month = "April"; break;
case 4: month = "May"; break;
case 5: month = "June"; break;
case 6: month = "July"; break;
case 7: month = "August"; break;
case 8: month = "September"; break;
case 9: month = "October"; break;
case 10: month = "November"; break;
default: month = "December"; break;
}
calendarString.append(month + " " + calendar.get(Calendar.DAY_OF_MONTH) + ", ");
calendarString.append(calendar.get(Calendar.YEAR) + "."); // First line ends here (Add Jobs this month)
calendarString.append(System.getProperty("line.separator"));
calendarString.append(System.getProperty("line.separator"));
calendarString.append(" Su ");
calendarString.append(" M ");
calendarString.append(" T ");
calendarString.append(" W ");
calendarString.append(" T ");
calendarString.append(" F ");
calendarString.append(" S ");
calendarString.append(System.getProperty("line.separator"));
答案 0 :(得分:1)
您不应该使用Calendar
& GregorianCalendar
课程。它们很麻烦,容易混淆,有缺陷,而且设计很差。现在legacy取代了java.time类。
LocalDate
LocalDate
类表示没有时间且没有时区的仅限日期的值。
时区对于确定日期至关重要。对于任何给定的时刻,日期在全球范围内因地区而异。例如,在Paris France午夜后的几分钟是新的一天,而Montréal Québec中仍然是“昨天”。
以continent/region
的格式指定proper time zone name,例如America/Montreal
,Africa/Casablanca
或Pacific/Auckland
。切勿使用诸如EST
或IST
之类的3-4字母缩写,因为它们不是真正的时区,不是标准化的,甚至不是唯一的(!)。
ZoneId z = ZoneId.of( "America/Montreal" );
LocalDate today = LocalDate.now( z );
要增加日期,只需一次添加一天。
LocalDate nextDay = today.plusDays( 1 );
DayOfWeek
您可以查询星期几作为DayOfWeek
枚举对象。
DayOfWeek dayOfWeek = today.getDayOfWeek();
您可以查询日期(日期编号)。
int dayOfMonth = today.getDayOfMonth();
您可以使用Month
枚举自动定位月份名称。因此,不需要在示例代码中看到该开关switch(theMonth)
。顺便说一句,请注意java.time在1月到12月的数字是1-12个月,而不是遗留的Calendar
类中看到的疯狂的0-11。
String monthName = today.getMonth().getDisplayName( TextStyle.FULL , Locale.CANADA_FRENCH );
我无法在一个月内打印正确的周数
无需知道周数。当您一次增加一天时,请检查第二天何时切换到下个月。一种简单的方法是跟踪作为YearMonth
对象打印的月份。
YearMonth ym = YearMonth.from( today );
…
YearMonth ymNextDay = YearMonth.from( nextDay );
if ( ymNextDay.isAfter( ym ) ) {
… print your next month name here
由于您只需要当月的更多月份,因此您需要确定结束月份。在你的情况下,将是下个月。您可以使用YearMonth
类进行此类数学运算。
YearMonth ymStop = ymStart.plusMonths( 1 ); // Go from first month to the following month.
解决问题的方法不止一种。替代方法确实可以计算周数。我怀疑这样做会使你的特定问题复杂化。
如何在一周中的正确日期启动日历
简单算一算。
如果第一个日期是星期四,并且您将星期日视为一周的第一天,那么您必须创建"空白"输出中前四个插槽(列)的(所有SPACE字符)。
如果您认为星期一是世界上常见的第一天和ISO 8601标准,您可以通过调用{{1}来询问DayOfWeek
订单中的每周订单从1到7获取一个值。如果不是星期一,那么你需要另一种方法来计算从一周开始的天数。您可以使用DayOfWeek::getValue
跟踪预期的星期几订单。
List
要求List中的位置返回从零开始的索引号。因此,添加一个以获得一周内的序数位置。
List<DayOfWeek> daysOfWeekInOrder = new ArrayList<>( 7 );
daysOfWeekInOrder.add( DayOfWeek.SUNDAY );
daysOfWeekInOrder.add( DayOfWeek.MONDAY );
…
daysOfWeekInOrder.add( DayOfWeek.SATURDAY );
星期四,您将获得int dayNumber = daysOfWeekInOrder.indexOf( today.getDayOfWeek() ) + 1 ;
,因此您知道需要循环5
次(4 = 5 - 1)才能生成空白。
同样,如果您的月份的最后一天是星期一,则可以询问星期一的天数,获取4
,从2
减去7
,这样您就知道了你需要产生五个空白来完成这个月。
java.time框架内置于Java 8及更高版本中。这些类取代了麻烦的旧legacy日期时间类,例如java.util.Date
,Calendar
和&amp; 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。