1 Java.util.Date
包含年、月、日、时、分、秒信息。
// String转换为Date
String dateStr="2013-8-13 23:23:23";
String pattern="yyyy-MM-dd HH:mm:ss";
DateFormate dateFormat=new SimpleDateFormat(pattern);
Date date=dateFormat.parse(dateStr);
date=dateFormat.format(date);
2 Java.sql.Date
包含年、月、日信息。
继承自java.util.Date。在数据库相关操作中使用,如rs.getDate,ps.setDate等。rs是指ResultSet,ps是指PreparedStatement。
// java.util.Date转换为java.sql.Date
new java.sql.Date(utilDate.getTime());// 其中utilDate为java.util.Date类型的对象
3 Java.util.Calendar
包含年、月、日、时、分、秒、毫秒信息。
JDK1.1引入,用以代替java.util.Date。
// Date转为Calendar
Date date=new Date();
Calendar calendar=Calendar.getInstance();
calendar.setTime(date);
// Calendar转为Date
Calendar ca=Calendar.getInstance();
Date d =(Date) ca.getTime();
4 Java.sql.Timestamp
包含年、月、日、时、分、秒、 纳秒(nano)信息。
继承自java.util.Date。比java.sql.Date包含更多信息。在数据库相关操作中使用,如rs.getTimestamp,ps.setTimeStamp等。例如:若数据库中某字段hireDate为Oracle的Date类型,则使用getTimestamp时能够将年、月、日、时、分、秒信息取出;但使用getDate时则只能取出年、月、日信息。因此,一般推荐使用getTimestamp。
// java.util.Calendar转换为java.sql.Timestamp
new Timestamp(Calendar.getInstance().getTimeInMillis());
// java.util.Date转换为java.sql.Timestamp
new Timestamp(date.getTime());
// String转换为java.sql.Timestamp,String格式:yyyy-mm-dd hh:mm:ss[.f...] ,方括号表示可选
Timestamp.valueOf("2013-07-06 01:49:30");
5 Oracle数据库提供的日期和时间类型
Oracle数据库提供了DATE,TIMESTAMP,TIMESTAMP WITH TIME ZONE和TIMESTAMP WITH LOCAL TIME ZONE四种类型。
DATE包含世纪、年、月、日、时、分、秒信息。
TIMESTAMP是DATE的扩展,包含年、月、日、时、分、秒和fractional seconds信息。定义TIMESTAMP的格式如下:
TIMESTAMP [(fractional_seconds_precision)]
// 格式
TIMESTAMP 'YYYY-MM-DD HH24:MI:SS.FF'
// 一个例子
TIMESTAMP '1997-01-31 09:26:50.12'
其中fractional_seconds_precision是可选的,用于指定秒使用含几位小数的浮点数表示,它的取值范围是0到9,默认是6。上述例子中表示采用两位小数,它的秒值是50.12。注意:12不是毫秒值,也不是微秒值。
Java 的开发过程中免不了与 Date 类型纠缠,准备总结一下项目经常使用的日期相关操作,JDK 版本 1.7,如果能够帮助大家节约那么几分钟起身活动一下,去泡杯咖啡,便是极好的,嘿嘿。当然,我只提供了可行的解决方案,并不保证是最佳实践,欢迎讨论。
1. 日期取值
在旧版本 JDK 的时代,有不少代码中日期取值利用了 java.util.Date 类,但是由于 Date 类不便于实现国际化,其实从 JDK1.1 开始,就更推荐使用 java.util.Calendar 类进行时间和日期方面的处理。这里便不介绍 Date 类的操作,让我们直奔主题吧,如何利用 Calendar 类取得现在的日期时间。
由于 Calendar 的构造器方法被 protected 修饰,所以我们会通过 API 中提供的 getInstance 方法来创建 Calendar 对象。
1 //有多个重载方法创建 Calendar 对象 2 Calendar now = Calendar.getInstance(); //默认 3 //指定时区和地区,也可以只输入其中一个参数 4 Calendar now = Calendar.getInstance(timeZone, locale);
然后我们就可以通过该对象取得当前的各种时间参数了。
int year = now.get(Calendar.YEAR); //2015,当前年份 int month = now.get(Calendar.MONTH) + 1; //12,当前月,注意加 1 int day = now.get(Calendar.DATE); //23,当前日 Date date = now.getTime(); //直接取得一个 Date 类型的日期
要取得其他类型的时间数据仅需修改 now.get() 内的参数,除了以上三种参数,其他常用参数如下:
- Calendar.DAY_OF_MONTH:日期,和 Calendar.DATE 相同
- Calendar.HOUR:12 小时制的小时数
- Calendar.HOUR_OF_DAY:24小时制的小时数
- Calendar.MINUTE:分钟
- Calendar.SECOND:秒
- Calendar.DAY_OF_WEEK:周几
除了取得时间数据,我们也可以通过 Calendar 对象设置各种时间参数。
1 //只设定某个字段的值 2 // public final void set(int field, int value) 3 now.set(Calendar.YEAR, 2016); 4 //设定年月日或者年月日时分或年月日时分秒 5 // public final void set(int year, int month, int date[, int hourOfDay, int minute, int second]) 6 now.set(2016, 1, 1[, 11, 1, 1]); 7 //直接传入一个 Date 类型的日期 8 // public final void setTime(Date date) 9 now.set(date);
注意:
- 当设置了时间参数后,其他相关的数值都会重新计算,例如当你把日期设为 11 号后,周几就会作对应变化。
- 获得的月份加 1 才是实际月份。
- 在 Calendar 类中,周日是 1,周一是 2,以此类推。
2. 日期转换
聊完日期取值,接下来聊聊日期转换,转换一般是 Date 型日期与 String 型字符串之间的相互转换,我主要利用 java.text.SimpleDateFormat 进行转换操作。
1 SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); 2 try { 3 //日期转字符串 4 Calendar calendar = Calendar.getInstance(); 5 Date date = calendar.getTime(); 6 String dateStringParse = sdf.format(date); 7 //字符串转日期 8 String dateString = "2016-01-01 11:11:11"; 9 Date dateParse = sdf.parse(dateString); 10 } catch (ParseException e) { 11 e.printStackTrace(); 12 }
注意:
- 创建 SimpleDateFormat 对象时必须指定转换格式。
- 转换格式区分大小写,yyyy 代表年份,MM 代表月份,dd 代表日期,HH 代表 24 进制的小时,hh 代表 12 进制的小时,mm 代表分钟,ss 代表秒。
3. 日期加减
通常来说,我们会对日期做两种加减操作:
- 以某个日期为基准,计算其几天前/后、几年前/后,或者其他时间单位前后的日期
按 Ctrl+C 复制代码按 Ctrl+C 复制代码
注意使用了 Calendar 对象的 add 方法,可以更改 Calendar.YEAR 为任意时间单位字段,完成各种时间单位下的日期计算。
- 计算两个时间的间隔,例如计算 2016 年 1 月 1 日距离现在有多少天。
1 SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); 2 String dateString = "2016-01-01 11:11:11"; 3 Calendar calendar = Calendar.getInstance(); 4 long nowDate = calendar.getTime().getTime(); //Date.getTime() 获得毫秒型日期 5 try { 6 long specialDate = sdf.parse(dateString).getTime(); 7 long betweenDate = (specialDate - nowDate) / (1000 * 60 * 60 * 24); //计算间隔多少天,则除以毫秒到天的转换公式 8 System.out.print(betweenDate); 9 } catch (ParseException e) { 10 e.printStackTrace(); 11 }
4. 日期比较
翻看自己以前的代码,发现每当进行日期比较的操作时,总会先将日期转为 “yyyyMMdd” 格式的字符串,再将字符串转为数值,然后比较数值大小。哈哈,一个简单的比较操作,却要写十几行代码,有点目不忍视。现在得说说正确地日期比较姿势是怎么样的。
日期比较一般有两种方法,对于 java.util.Date 或者 java.util.Calendar 都是通用的。一种是通过 after() 与 before() 方法进行比较,一种是通过 compareTo() 方法进行比较。
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); String dateString_01 = "2016-01-01 11:11:11"; String dateString_02 = "2016-01-02 11:11:11"; try { Date date_01 = sdf.parse(dateString_01); Date date_02 = sdf.parse(dateString_02); System.out.println(date_01.before(date_02)); //true,当 date_01 小于 date_02 时,为 true,否则为 false System.out.println(date_02.after(date_01)); //true,当 date_02 大于 date_01 时,为 true,否则为 false System.out.println(date_01.compareTo(date_02)); //-1,当 date_01 小于 date_02 时,为 -1 System.out.println(date_02.compareTo(date_01)); //1,当 date_02 大于 date_01 时,为 1 System.out.println(date_02.compareTo(date_02)); //0,当两个日期相等时,为 0 } catch (ParseException e) { e.printStackTrace(); }
5. 工具库推荐
看见有朋友向我推荐了 Joda-Time 库,经过研究后发现能够简化不少 Java 中日期时间的操作,特别是在某些应用场景下还得 Date 转 Calendar 或者 String 转 Date 之类的操作。库的使用逻辑与原生十分相像,学习曲线还是比较平缓的,在此也向各位读者提供一个更加多元的选择。
Joda-Time 官方网站:http://www.joda.org/joda-time/
都是常用的日期之间的比较方法,供参考。
SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");//设置日期格式
String nowDate = df.format(new Date());// new Date()为获取当前系统时间
注:楼主一直以为date类型不如string类型随心所欲,所以下面的比较都是string类型的date进行比较。如果你真的真的就是那么倔强,ok,下面是date转string的方法:
SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");//设置日期格式
String date = df.format(Date类型的时间);
1.两个string类型的日期比较大小
1 public static int compare_date(String DATE1, String DATE2) { 2 DateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); 3 try { 4 Date dt1 = df.parse(DATE1); 5 Date dt2 = df.parse(DATE2); 6 if (dt1.getTime() > dt2.getTime()) { 7 System.out.println("dt1 在dt2前"); 8 return 1; 9 } else if (dt1.getTime() < dt2.getTime()) { 10 System.out.println("dt1在dt2后"); 11 return -1; 12 } else { 13 return 0; 14 } 15 } catch (Exception exception) { 16 exception.printStackTrace(); 17 } 18 return 0; 19 }
2.返回两个string类型日期之间相差的天数
1 public static int daysBetween(String smdate,String bdate){ 2 SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd"); 3 Calendar cal = Calendar.getInstance(); 4 long time1 = 0; 5 long time2 = 0; 6 7 try{ 8 cal.setTime(sdf.parse(smdate)); 9 time1 = cal.getTimeInMillis(); 10 cal.setTime(sdf.parse(bdate)); 11 time2 = cal.getTimeInMillis(); 12 }catch(Exception e){ 13 e.printStackTrace(); 14 } 15 long between_days=(time2-time1)/(1000*3600*24); 16 17 return Integer.parseInt(String.valueOf(between_days)); 18 }
3.返回两个string类型日期相差的小时数
1 public static int daysBetween2(String startTime, String endTime) { 2 SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd HH"); 3 Calendar cal = Calendar.getInstance(); 4 long time1 = 0; 5 long time2 = 0; 6 7 try{ 8 cal.setTime(sdf.parse(startTime)); 9 time1 = cal.getTimeInMillis(); 10 cal.setTime(sdf.parse(endTime)); 11 time2 = cal.getTimeInMillis(); 12 }catch(Exception e){ 13 e.printStackTrace(); 14 } 15 long between_days=(time2-time1)/(1000*3600); 16 17 return Integer.parseInt(String.valueOf(between_days)); 18 }
4.计算两段日期的重合日期
1 /** 2 * 计算两段日期的重合日期 3 * @param str1 开始日期1 4 * @param str2 结束日期1 5 * @param str3 开始日期2 6 * @param str4 结束日期2 7 * @return 8 * @throws Exception 9 */ 10 public static Map<String,Object> comparisonRQ(String str1, String str2, String str3, 11 String str4) throws Exception { 12 String mesg = ""; 13 DateFormat df = new SimpleDateFormat("yyyy-MM-dd"); 14 String startdate = ""; 15 String enddate = ""; 16 try { 17 Date dt1 = df.parse(str1); 18 Date dt2 = df.parse(str2); 19 Date dt3 = df.parse(str3); 20 Date dt4 = df.parse(str4); 21 if (dt1.getTime()<=dt3.getTime()&&dt3.getTime()<=dt2.getTime()&&dt2.getTime()<=dt4.getTime()) { 22 mesg = "f";//重合 23 startdate = str3; 24 enddate = str2; 25 } 26 if (dt1.getTime()>=dt3.getTime()&&dt3.getTime()<=dt2.getTime()&&dt2.getTime()<=dt4.getTime()) { 27 mesg = "f";//重合 28 startdate = str1; 29 enddate = str2; 30 } 31 32 if (dt3.getTime()<=dt1.getTime()&&dt1.getTime()<=dt4.getTime()&&dt4.getTime()<=dt2.getTime()) { 33 mesg = "f";//重合 34 startdate = str1; 35 enddate = str4; 36 } 37 if (dt3.getTime()>=dt1.getTime()&&dt1.getTime()<=dt4.getTime()&&dt4.getTime()<=dt2.getTime()) { 38 mesg = "f";//重合 39 startdate = str3; 40 enddate = str4; 41 } 42 43 System.out.println(startdate+"----"+enddate); 44 45 46 }catch (ParseException e) { 47 e.printStackTrace(); 48 throw new ParseException(e.getMessage(), 0); 49 }catch(Exception e){ 50 e.printStackTrace(); 51 throw new Exception(e); 52 } 53 Map<String,Object> map = new HashMap<String,Object>(); 54 map.put("startdate", startdate); 55 map.put("enddate", enddate); 56 return map; 57 }