java时间操作三部曲之java.util.Date
2016年8月来到一家新公司,认识了现在的CTO彪哥,也许和很多初出茅庐的程序员一样,见到技术大神内心膜拜之情溢于言表。我也一样,想跟着学这学那,不断总结自己的不足,向彪哥请教。 写博客的想法就是从彪哥那学的。
第一次写,也不知道写些什么,我给自己的定位是:一·自己的学习笔记,方便以后温习和工具查询 二·也算是满足自己的小虚荣吧,总觉得很多年以后在看现在写的博客,会有一种满足感。所以这一次,酝酿了很久打算写一写java中关于时间的操作,重新学习总结一下Date,Calendar,GregorienCalendar这三个类。
今天这一篇就写java.util.Date
java.util.Date 类代表某一特定的时间,精确到毫秒。该类中所有方法接受或返回的年,月,日,时,分,秒和秒值都遵循以下规则
YEAR(年)由一个整数表示y-1900(等会解释为什么要-1900)
MONTH(月)由一个整数表示从0-11,也就是说0表示0+1月,即0表示1月,1表示2月以此类推
DAY(月日)由一个整数表示 1-31
HOUR(小时)从0-23和年类似,0表示0+1点
MINUTE(分)从0-59
类构造函数
1.Date() 这个构造函数分配一个Date对象并将它初始化,使它表示其被分配的时间,精确到毫秒。
java.util.Date date = new java.util.Date();
2.Date(String s) 接受一个时间字符串,设定某个时间,jdk1.1后不推荐使用了。
java.util.Date date = new java.util.Date("2016-10-26");
3.Date(long date)
这个构造函数分配一个Date对象并初始化它代表指定的毫秒数,因为被称为“纪元”,即1970年1月1日00:00:00 GMT标准基准时间。
4.以上是几个比较常用的构造函数,还有一些不常用的,并且在jdk1.1后不被推荐使用的,如:Date(int year, int month, int date),Date(int year, int month, int date, int hrs, int min)等。在使用上面两个构造函数的时候 year 参数需要理想年-1900,例如2016年,作为参数用的话需要先减去1900年,也就是:
java.util.Date date = new java.util.Date (116,10,26); //Date的源代码如下 @Deprecated public Date(int year, int month, int date, int hrs, int min, int sec) { int y = year + 1900;
类方法
1.boolean after(Date when) 此方法测试,此日期是否在指定日期之后。参数 when --进行检查的日期;返回值:true表示代表Date对象是延迟与when表示的时刻,否则false
Date begin = new Date(116,10,26); Date end = new Date(116,11,26); System.out.println(begin.after(end));////输出false 表示begin在end之前,end比begin晚
2.boolean before(Date when) 此方法测试,此日期是否在指定日期之前。和上面的方法相反
Date begin = new Date(116,10,26); Date end = new Date(116,11,26); System.out.println(begin.before(end));//输出true 表示begin在end之前,早于end
3.Object clone() 返回此Date对象的浅表副本。
Date begin = new Date(116,10,26); Date clone=(Date) begin.clone(); System.out.println(begin);//Sat Nov 26 00:00:00 CST 2016 System.out.println(clone);//Sat Nov 26 00:00:00 CST 2016
2016-10-27补充:clone方法,在创建实体类的时候可能会用到,java核心技术卷1中有个警告:不要编写返回引用可变对象的访问器(getXX()方法)。什么意思呢,看下面代码
public class Employee { private Date hireDay;//雇佣时间 public Date getHireDay() { return hireDay; } public void setHireDay(Date hireDay) { this.hireDay = hireDay; } public static void main(String[] args) { //下面的代码就会破坏封装性 Employee e=new Employee(); e.setHireDay(new Date()); System.out.println(e.getHireDay());//Thu Oct 27 10:09:05 CST 2016 Date d=e.getHireDay(); //因为e对象的hireDay和变量d引用同一个对象,一个修改了,另一个也会被修改 d.setTime(12345768999999L); System.out.println(e.getHireDay());//Thu Mar 23 04:16:39 CST 2361 } }
修改一下get,set方法
public Date getHireDay() { return (Date) hireDay.clone(); } public void setHireDay(Date hireDay) { this.hireDay = (Date) hireDay.clone(); }
4.int compareTo(Date anotherDate)方法比较两个日期。 参数:anotherDate要比较的日期;返回值:0如果参数日期等于此日期; 如果这个日期在Date参数之前返回一个小于0的值 ; 如果这个日期在Date参数之后返回一个大于0的值。
Date date = new Date(98, 5, 21); Date date2 = new Date(99, 1, 9); // make 3 comparisons with them int comparison = date.compareTo(date2); int comparison2 = date2.compareTo(date); int comparison3 = date.compareTo(date); System.out.println("Comparison Result:" + comparison);//-1 System.out.println("Comparison2 Result:" + comparison2);//1 System.out.println("Comparison3 Result:" + comparison3);//0
5.long getTime() 方法返回自1970年1月1日00:00:00 GMT已经过去了多少毫秒,返回一个long类型的时间戳
Date begin = new Date(116,10,26); System.out.println(begin.getTime());//1480089600000
6.boolean equals(Object obj) 方法检查如果两个日期都是相等的,基于毫秒的差异。这个方法其实就是比较的两个时间的long类型的时间戳是否相等,看源码
public boolean equals(Object obj) { return obj instanceof Date && getTime() == ((Date) obj).getTime(); }
常用的方法也就这些,另外setTime(long time)是给他一个时间戳设置时间的。toString()方法是重写了object的方法。
常见问题
1.比较时间的早晚
如果比较连个时间的早晚的话,建议使用原生态的Date的befor和after方法,效率相对要高,通过分析源码,我们可以知道,其他就是比较的两个long类型的数字
public boolean before(Date when) { return getMillisOf(this) < getMillisOf(when); } static final long getMillisOf(Date date) { if (date.cdate == null || date.cdate.isNormalized()) { return date.fastTime; } BaseCalendar.Date d = (BaseCalendar.Date) date.cdate.clone(); return gcal.getTime(d); }
2.时间格式化
Date类型的时间格式话,需要借助java.text.SimpleDateFormat("yyyy-MM-dd hh:mm:ss")对象,后面是要输出的格式
/** 输出格式: 2016-10-26 4:48:11*/ System.out.println((new java.text.SimpleDateFormat("yyyy-M-d h:mm:ss")).format(new Date())); /** 输出格式: 2016-10-26 16:48:11 */ System.out.println((new java.text.SimpleDateFormat("yyyy-M-d H:mm:ss")).format(new Date()));
这里需要注意的就是24时和12时的显示正则,H大写为24时计时法,h小写为12时计时法,两个h时,即hh,当为个位数时间时前面自动补0即下午4点显示04
3.还有时间加减天数,建议借助工具类,后面会讲到。但是这里呢还是要讲一个问题,对于有些刚接触java的朋友,在计算时间增减的时候喜欢用
System.out.println(new java.util.Date().getTime()-1000*60*60*24*30); //计算一个月前的日期
这个时候得到的结果是不正确的,因为1000*60*60*24*30 这个计算后默认是int类型,但是实际的值已经超出了int类型存储的长度,转成long类型,万事大吉。
4.还有一个是夏令时问题,原来Java中不是每天都是标准的24个小时,可能是23,也可能是25,日期的计算,使用Calendar提供的API,是不会出差错的,简单的new Date(long milliseconds)可能并不靠谱
参考:http://www.cnblogs.com/snake-hand/archive/2013/06/10/3131157.html
5.java.util.Date 和 java.sql.Date的区别和相互转化
util包下的Date是在除SQL语句外使用的,sql包下的Date是针对SQL语句中使用,它只有日期而没有时间部分。
// java.sql.Date转为java.util.Date java.sql.Date date=new java.sql.Date(); java.util.Date d=new java.util.Date (date.getTime()); // java.util.Date转为java.sql.Date java.util.Date utilDate=new Date(); java.sql.Date sqlDate=new java.sql.Date(utilDate.getTime());
6.还有就是时区导致new Date()获取的时间缺时少分,可以查看一下系统默认时间,当然后面提供的Calendar类都有了很好的改善
参考:http://blog.csdn.net/yohoph/article/details/7601377
第一次写,先写这么多吧,让我对Date又有了一个新的认识。希望看到的朋友,对date有更高的认识或者对java中时间操作有更好的建议的,可以留言,多交流。2016-10-26