【总结】java.util.Date vs. java.sql.Date

本文总结了java.util.Date和java.sql.Date的不同。非原创,是基于stack overflow的问答的总结。
原贴:[url]http://stackoverflow.com/questions/2305973/java-util-date-vs-java-sql-date[/url]


类图如下图如示,java.sql.Date 是java.util.Date的子类。
[img]http://dl.iteye.com/upload/picture/pic/135321/194ba9da-c16f-3717-9f7e-1f973591b09c.png[/img]

一般来说,数据库通常支持三种以上的时间格式,即:DATE, TIME,以及TIMESTAMP。每种都对应着一个JDBC类,即:java.sql.Date,java.sql.Time,java.sql.Timestamp。而以上三个类都继承了java.util.Date类。

1、 java.sql.Date对应着SQL的DATE(即:'YYYY-MM-DD'),表示年月日。(没错,是没有时分秒的)。此外,sql.Date没有时区。
2、 java.sql.Time对应着SQL的TIME,它只包含了时分秒(即:'HH:MM:SS'),以及毫秒信息。
3、 java.sql.Timestamp对应着SQL的TIMESTAMP(时间戳),支持纳秒。(值得注意的是:util.Date只支持毫秒。)

[table]
|java.sql.Date|DATE|YYYY-MM-DD|
|java.sql.Time|TIME|HH:MM:SS|
|java.sql.Timestamp|TIMESTAMP||
[/table]

最常见的一个bug是用在用与数据库相关的这三种类型(即java.sql.*)时,没有正确的使用,比如sql.Date是无时区的,sql.Time没有年月日的。

[b]最后:要怎样选择?[/b]
这个取决于数据库的列类型。 PreparedStatement有所有类型的setter方法。即:
[table]
|PreparedStatement#setDate()|sql.Date|
|PreparedStatement#setTime()|sql.Time|
|PreparedStatement#setTimestamp()|sql.Timestamp|
[/table]

注意:如果是使用ps.setObject(fieldIndex, utilDateObject); 第二个参数可以传入util.Date类型,因为大多数的JDBC驱动可以识别并转成相对应正确的类型。再次取出的时候,需要注意的是可能会丢失掉一些信息(比如DATE类型,拿出的时候会丢掉时分秒信息)。

[b]建议:不要使用以上任何一种Date类型。[/b]
比较推荐的做法是存放时间格式的时候用毫秒或纳秒格式存放(long数据类型),然后在取出的时候使用joda-time(http://www.joda.org/joda-time/)转成想要的类型。

[b]解释:[/b]
[b]时间戳(Timestamp)[/b]:指格林威治时间(Greenwich Mean Time,GMT, 零时区)1970年01月01日00时00分00秒(北京时间1970年01月01日08时00分00秒)起至现在的总秒数。编程人员对于这个肯定不陌生,是非常常见的时间类。

[b]Joda-Time[/b]:提供了高质量的替代java date和time类的方法。在JAVA SE8官方文档中已经纳入了Joda-Time。

-------------------------------------------
参考:
[url]http://docs.oracle.com/javase/6/docs/api/java/sql/Date.html[/url]
[url]http://www.joda.org/joda-time[/url]
[url]http://h819.iteye.com/blog/611099[/url]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值