接触java不久,感觉java真的挺好玩的。
Calendar
类是一个抽象类,它为特定瞬间与一组诸如 YEAR
、MONTH
、DAY_OF_MONTH
、HOUR
等日历字段之间的转换提供了一些方法,并为操作日历字段(例如获得下星期的日期)提供了一些方法。
在书上看到一个挺好玩的代码,通过这个代码浅浅的研究了下java中的Calendar类
输出的结果是:
PS:为了学习和理解方便,我将时间调回2011年,因为2012年刚好1月1号刚好是周日。
字段操作:
三种:set(),add(),roll();
set(f, value)
将日历字段f
更改为value
。此外,它设置了一个内部成员变量,以指示日历字段f
已经被更改。尽管日历字段f
是立即更改的,但是直到下次调用get()
、getTime()
、getTimeInMillis()
、add()
或roll()
时才会重新计算日历的时间值(以毫秒为单位)。因此,多次调用set()
不会触发多次不必要的计算。使用set()
更改日历字段的结果是,其他日历字段也可能发生更改,这取决于日历字段、日历字段值和日历系统。此外,在重新计算日历字段之后,get(f)
没必要通过调用set
方法返回value
集合。具体细节是通过具体的日历类确定的。
示例:假定
GregorianCalendar
最初被设置为 1999 年 8 月 31 日。调用set(Calendar.MONTH, Calendar.SEPTEMBER)
将该日期设置为 1999 年 9 月 31 日。如果随后调用getTime()
,那么这是解析 1999 年 10 月 1 日的一个暂时内部表示。但是,在调用getTime()
之前调用set(Calendar.DAY_OF_MONTH, 30)
会将该日期设置为 1999 年 9 月 30 日,因为在调用set()
之后没有发生重新计算。
个人建议先跳过规则看下示例,然后再回过来理解。
add(f, delta)
将delta
添加到f
字段中。这等同于调用set(f, get(f) + delta)
,但要带以下两个调整:Add 规则 1。调用后
f
字段的值减去调用前f
字段的值等于delta
,以字段f
中发生的任何溢出为模。溢出发生在字段值超出其范围时,结果,下一个更大的字段会递增或递减,并将字段值调整回其范围内。Add 规则 2。如果期望某一个更小的字段是不变的,但让它等于以前的值是不可能的,因为在字段
f
发生更改之后,或者在出现其他约束之后,比如时区偏移量发生更改,它的最大值和最小值也在发生更改,然后它的值被调整为尽量接近于所期望的值。更小的字段表示一个更小的时间单元。HOUR
是一个比DAY_OF_MONTH
小的字段。对于不期望是不变字段的更小字段,无需进行任何调整。日历系统会确定期望不变的那些字段。此外,与
set()
不同,add()
强迫日历系统立即重新计算日历的毫秒数和所有字段。
示例:假定
GregorianCalendar
最初被设置为 1999 年 8 月 31 日。调用add(Calendar.MONTH, 13)
将日历设置为 2000 年 9 月 30 日。Add 规则 1 将MONTH
字段设置为 September,因为向 August 添加 13 个月得出的就是下一年的 September。因为在GregorianCalendar
中,DAY_OF_MONTH
不可能是 9 月 31 日,所以 add 规则 2 将DAY_OF_MONTH
设置为 30,即最可能的值。尽管它是一个更小的字段,但不能根据规则 2 调整DAY_OF_WEEK
,因为在GregorianCalendar
中的月份发生变化时,该值也需要发生变化。
以下是代码,书上的代码及注释+我自己的理解和调试观察。
import java.text.DateFormatSymbols; import java.util.*; /** * @version 1.4 2007-04-07 * @author Cay Horstmann */ public class CalendarTest { public static void main(String[] args) { // construct d as current date //GregorianCalendar 是 Calendar 的一个具体子类,提供了世界上大多数国家/地区使用的标准日历系统。 //用来表示默认地区。默认时区的当前时间。 GregorianCalendar d = new GregorianCalendar(); int today = d.get(Calendar.DAY_OF_MONTH); int month = d.get(Calendar.MONTH);//0为第一个月 //以下一段代码是测试,类里面的一些方法,测试娱乐。 /* * int hour = d.get(Calendar.HOUR_OF_DAY);//0~23 * int minute = d.get(Calendar.MINUTE); //0~59 * int second = d.get(Calendar.SECOND); //0~59 * System.out.println(month+1+"月"+today+"日"+hour+"时"+minute+"分"+second+"秒"); */ // set d to start date of the month d.set(Calendar.DAY_OF_MONTH, 1); //Sunday==1; 1<=weekday<=7; int weekday = d.get(Calendar.DAY_OF_WEEK); //因为是周六,所以weekday == 7; //System.out.println(weekday); // get first day of week (Sunday in the U.S.) int firstDayOfWeek = d.getFirstDayOfWeek(); //firstDayOfWeek==1; // determine the required indentation for the first line int indent = 0; //一周的第一天是否是月的第一天,如果不是,则追溯到上个月找到这个周的第一天所在的时间。 while (weekday != firstDayOfWeek) { indent++; d.add(Calendar.DAY_OF_MONTH, -1); weekday = d.get(Calendar.DAY_OF_WEEK); //int temp=d.get(Calendar.DAY_OF_MONTH); //System.out.println(temp); } /* * DateFormatSymbols 是用于压缩本地化的日期_时间格式化数据,如月份名称、星期名称和时区数据的公有类。 * * public String[] getShortWeekdays() * 获得短型工作日字符串。 例如:"Sun", "Mon" 等。 * 返回值:短型工作日字符串。 */ // print weekday names String[] weekdayNames = new DateFormatSymbols().getShortWeekdays(); /* * 以下循环操作类似于这个,但是前面一系列操作用于确定第一天是从周几开始(因地区而异)。所以不用我下面这两行程序 * for(int i=1; i<=7; i++) * System.out.printf("%s", weekdayNames[i]); * * 此时:weekday == firstDayOfWeek * 但地下循环为先执行后判断,即从周的选定的第一天 开始循环7次 */ do { System.out.printf("%4s", weekdayNames[weekday]); d.add(Calendar.DAY_OF_MONTH, 1); weekday = d.get(Calendar.DAY_OF_WEEK); }while (weekday != firstDayOfWeek); System.out.println(); //月的1号一般不是从周的第一天开始。比如:周的第一天是周日,而2011年的第一天是周六,空出前6天的位置。 for (int i = 1; i <= indent; i++) System.out.print(" "); d.set(Calendar.DAY_OF_MONTH, 1); do { // print day int day = d.get(Calendar.DAY_OF_MONTH); System.out.printf("%3d", day); // mark current day with * if (day == today) System.out.print("*"); else System.out.print(" "); // advance d to the next day d.add(Calendar.DAY_OF_MONTH, 1); weekday = d.get(Calendar.DAY_OF_WEEK); // start a new line at the start of the week if (weekday == firstDayOfWeek) System.out.println(); }while (d.get(Calendar.MONTH) == month); // the loop exits when d is day 1 of the next month // print final end of line if necessary if (weekday != firstDayOfWeek) System.out.println(); } }