记得在我刚学Java的时候,真是搞不清楚Date和Calendar这两个类,后来我渐渐知道,原来不能全怪我啊,Java日期API之烂是公认的(不妨参见这篇文章,Tiago Fernandez做过一个投票,就是要选举最烂的Java API,结果Java日期API排行第二,仅次于臭名远扬的EJB2,嘿嘿)。
蛋疼的java.sql.Date
只有Date和Calendar搞定一切吗?那还好啊。当然不是!光Date就有java.util.Date和java.sql.Date,而且关系是java.sql.Date extends java.util.Date。为了把前者转成后者,我写了这样的代码:
1
2
|
Date date = new Date();
java.sql.Date d = new java.sql.Date(date.getTime());
|
居然不支持Date参数的构造器,我只好传入long类型的时间。接下去,我尝试把当前小时数取出来:
1
|
System.out.println(d.getHours());
|
悲剧出现了:
1
2
|
Exception in thread "main" java.lang.IllegalArgumentException
at java.sql.Date.getHours(Date.java:177)
|
一看源码,坑爹啊:
1
2
3
|
public int getHours() {
throw new java.lang.IllegalArgumentException();
}
|
在java.util.Date里面好好的方法怎么变成这个鸟样了?
方法注释给出了说明:
This method is deprecated and should not be used because SQL Date values do not have a time component.
也就是说,java.sql.Date是SQL中的单纯的日期类型,哪会有时分秒啊?我觉得它根本不应该设计成java.util.Date的子类。如果你把java.sql.Date通过JDBC插入数据库,你会发现时分秒都丢失了,因此如果你同时需要日期和时间,你应该使用Timestamp,它也是java.util.Date的子类。
另外还有一个java.util.Date的子类叫Time,java.sql包下面的Date、Time和Timestamp可以放在一起记忆。Date只包含年月日信息、Time只包含时分秒信息,而Times则包含时间戳的完整信息。
现在知道人家抛出IllegalArgumentException的用心良苦了吧……
坑爹的year和month
看看Date类的构造器:
1
|
public Date( int year, int month, int day)
|
长得并不奇葩嘛。
好,现在我要输出2012年的1月1号了:
1
2
|
Date date = new Date( 2012 , 1 , 1 );
System.out.println(date);
|
结果,你傻眼了:
1
|
Thu Feb 01 00:00:00 CST 3912
|
等等,这是啥?3192年?
原来实际年份是要在你的年份参数上加上个起始年份1900。
更坑爹的是,月份参数我不是给了1吗?怎么输出二月(Feb)了?
Date里面的月份居然是用0~11表示的,换句话说,一月用0来表示,二月用1来表示。如果不用常量或者枚举,很容易踩到坑里去,对不对?
后来发现Go语言的time.Date方法,对于月份做了个恶心但是不容易坑人的处理(看奇葩的月份参数啊):
1
|
func Date(year int , month Month, day, hour, min, sec, nsec int , loc *Location)
|
我甚至怀疑Google这样处理是在用极端的方法鄙视Java(另,据我所知,JavaScript好像也是这样的,月份从0开始)……
尝试Joda吧
最开始的时候,Date既要承载日期信息,又要做日期之间的转换,还要做不同日期格式的显示,职责较繁杂,从JDK 1.1 开始,这三项职责分开了:
- 使用Calendar类实现日期和时间字段之间转换;
- 使用DateFormat类来格式化和分析日期字符串;
- 而Date只用来承载日期和时间信息。
原有Date中的相应方法已废弃。不过,无论是Date,还是Calendar,都用着太不方便了,这是API没有设计好的地方。
比如Calendar的getInstance方法,并未提供一个指定年月日和时分秒的重载方法,每次要指定特定的日期时间,必须先获取一个表示当前时间的Calendar实例,再去设值,比如:
1
2
3
|
Calendar c = Calendar.getInstance();
c.set( 2012 , 0 , 1 , 11 , 11 , 11 );
System.out.println(c.getTime());
|
注意上面代码中对于年份的传值——是的,和Date不一样的是,Calendar年份的传值不需要减去1900(当然月份的定义和Date还是一样),这种不一致真是让人抓狂!
打印:
1
|
Sun Jan 01 11:11:11 CST 2012
|
有很多开源库都在努力弥补Java的这一问题,比如Joda-Time,获取Calendar对象和设置时间完全可以合成一步完成:
1
|
DateTime dateTime = new DateTime( 2012 , 1 , 1 , 11 , 11 , 11 , 0 );
|
而且,一月份总是可以传1来表示了。
再如,如果要给上述时间增加3天再按格式输出的话,使用Joda更加便捷:
1
|
System. out .println(dateTime.plusDays(3).toString( "E MM/dd/yyyy HH:mm:ss" );
|
有兴趣的话请阅读此文,并下载Joda-Time使用。
JSR-310
众所周知Java的规范就是多、而且啰嗦,这帮老大们(Export Group中除了有Oracle的人,还有IBM、Google和RedHat的人)终于再也无法忍受Java那么烂的日期API了,于是就有了JSR-310(感兴趣的请移步),官方的描述叫做“This JSR will provide a new and improved date and time API for Java.”,目前的阶段还在“Early Draft Review 2”,有得等。
JSR-310将解决许多现有Java日期API的设计问题。比如Date和Calendar目前是可变对象,你可以随意改变对象的日期或者时间,而Joda就将DateTime对象设计成String对象一样地不可变,能够带来线程安全等等的好处,因此这一点也将被JSR-310采纳。
很多JSR规范都是在程序员的诋毁和谩骂声中萌芽的,然后会有开源项目来尝试解决Java的这些弊端,最后就轮到JSR就去抄他们的实现。除了新的日期API,再比如JCache(JSR-107),你知道它抄了多少EhCache的东西么……
文章系本人原创,转载请注明作者和出处(http://www.raychase.net)
注:本博客已经迁移到个人站点 http://www.raychase.net/ ,欢迎大家访问收藏,本ITEye博客在数日后将不再更新。
分享到:
相关推荐
这个"Java 8 全套API.zip"压缩包包含的文档详细地概述了这个版本的所有API,为开发者提供了丰富的参考信息。下面将深入探讨Java 8中的关键知识点。 1. **Lambda表达式**: Lambda表达式是Java 8最重要的特性之一,...
总结来说,这个“用java+swing实现的文件时间修改工具”项目涵盖了Java的基础知识,如文件操作、GUI设计以及事件处理,同时也涉及到高级特性,如NIO.2 API的使用。通过此项目,开发者可以深化对Java编程的理解,提高...
在日期和时间处理方面,Java 8引入了新的日期和时间API(java.time包),替代了传统的java.util.Date和Calendar类,提供了更加强大、直观且易用的API,使得日期和时间的计算与格式化变得更加简单。 在并发编程领域...
嵌入的时间显示则意味着程序会实时同步系统时间,在屏幕上展示日期和时间信息。这需要程序能够访问操作系统的时间API,获取并更新当前时间。在Windows系统中,可以使用WinAPI函数GetSystemTime或GetLocalTime来获取...
scratch少儿编程逻辑思维游戏源码-米克 demo.zip
scratch少儿编程逻辑思维游戏源码-萝卜男孩拯救世界.zip
scratch少儿编程逻辑思维游戏源码-酷忍者.zip
教育科技_微信小程序_二手交易平台_大学校园二手书与物品循环利用公益系统_风华读书人校园二手交易平台_基于C2C模式的校内闲置物品交易系统_支持多校区独立运营的二手书交易平台_包含
全新UI彩虹外链网盘系统源码前后端美化模板整站 模版文件.zip
maoxig_nonebot-plugin-ai-timetable_32152_1745865455265
少儿编程scratch项目源代码文件案例素材-足球顶尖高手.zip
少儿编程scratch项目源代码文件案例素材-作战基地.zip
少儿编程scratch项目源代码文件案例素材-云端之上 1-4名玩家.zip
scratch少儿编程逻辑思维游戏源码-魔幻之塔.zip
scratch少儿编程逻辑思维游戏源码-楼层酷跑.zip
scratch少儿编程逻辑思维游戏源码-圈.zip
少儿编程scratch项目源代码文件案例素材-纸片马里奥自定义战役.zip
少儿编程scratch项目源代码文件案例素材-自由下落.zip
少儿编程scratch项目源代码文件案例素材-阻击蜈蚣.zip
健康监测与疾病预防_脉搏波分析_六轴加速度传感器_生理参数融合_STC12硬件采集_Android数据处理_SpringBoot后端_MySQL数据库_MatlabPython实验