22

在数据库中保存日期时,我遇到了一种非常奇怪的行为。在我的(Linux centOS 6.2)服务器上,我使用 glassfish 应用程序服务器(3.1.1 - build 12)和 Java(1.7.0_09),该应用程序是用 Java + GWT 开发的,它使用 PostgreSQL 服务器(9.2.1)。在应用程序内部,有几个日期字段保存在数据库中。日期字段使用 datepicker (http://code.google.com/p/gwt-datepicker, r30)。

db 关系的日期属性是日期类型(不是时间戳)。一些日期前一天保存在数据库中。该问题仅发生在间隔之间的日期,例如 31.03.1968 和 27.10.1968 之间,这让我想到了某种夏令时问题。但由于它不会发生在 1969 年,例如,我不能很好地隔离问题。我试图找到问题发生的其他日期间隔。例如,如果我在应用程序中选择 19.05.1968,则在保存到数据库后,日期将保存为 18.05.1968。

奇怪的是,我在另一台服务器上有同一个应用程序的另一个实例,并且在相同的日期它们被正确保存。这让我认为问题可能取决于:

  • 玻璃鱼配置;
  • java(java.util.Date 实现?);
  • 我缺少某种服务器配置

我试图将我的服务器的所有可能配置设置为欧洲/罗马(我的时区),但没有。任何想法?我该如何解决或调查这个问题?

更新: 1968 年是闰年。这个问题也发生在 1972 年,又是闰年。总结:“日期保存前一天”问题发生在夏令时日期间隔的闰年。

创建日期对象的代码部分是:

Date d = dateField.getSelectedDate();
if (d != null) {
    txtVal = DateTimeFormat.getFormat("dd/MM/yyyy").format(d);
}

其中 dateField 声明为:

transient private DatePicker dateField;

包是org.zenika.widget.client.datePicker.DatePicker(前面提到的gwt-datepicker-r30),DateTimeFormat指的是com.google.gwt.i18n.shared.DateTimeFormat

接受答案后更新:

我使用了这个解决方法:创建日期时,我使用以下代码:

final long hours12 = 12L * 60L * 60L * 1000L;
Date d = new Date(d1.getTime() + hours12);
4

3 回答 3

9

只需将日期时间设置为 12:00(而不是默认的 0:00)就可以了。问题是 GWT 时区库不包括 1990 年之前的所有闰年,因此您将在服务器上得到错误的时间(因为该值是以时间戳的形式发送的,并且间隔一小时)。

顺便说一句:GWT 有一个内置的日期选择器,在http://gwt.google.com/samples/Showcase/Showcase.html#!CwDatePicker查看它的演示

于 2012-11-08T09:40:35.970 回答
1

您使用的是 java.util.date 还是 java.sql。日期(后者是正确的)?我对 SQL Server 也有类似的问题,即使它是正常的,是的,与夏季有关。

基本上,您将 Java 日期存储为特定日期的午夜。如果是夏令时,日期会移到前一天的 23:00,然后时间会被截断!如果您发送带有“随机”时间戳的日期,您将在夏季 24 次中遇到此问题。

我不记得确切的解决方案(它不会直接帮助您,因为它指的是不同的 DBMS),但是 dB 中有一个设置说“在您收到日期时存储它”。您可以通过将 dbcolumn 更改为 timestamp 并查看时间的存储方式来测试是否是这种情况。

我做了一些研究 - 似乎 gwt-datepicker 有很多问题!http://code.google.com/p/google-apps-script-issues/issues/detail?id=2022 http://code.google.com/p/google-apps-script-issues/issues/detail ?id=2001

如果他们的计算在闰年有错误,我不会感到惊讶。此外,完全有可能只是您正在做的事情与您认为自己正在做的事情之间不匹配 - 使用 Dates 是一件非常困难的事情

要检查,请尝试:

final java.util.Date ud = dateField.getSelectedDate();;
final java.sql.Date sd = new java.sql.Date(ud.getTime());
System.out.println(ud);// this is what you pick from the DatePicker
System.out.println(sd);// this is what will be stored on the database

看看他们在闰年的夏天是否匹配。如果是 GWT 错误,http://code.google.com/p/google-apps-script-issues/是报告它的正确位置

于 2012-11-08T01:25:00.560 回答
0

出生日期实时面对这个问题。

我进行了以下修复,因此不是在上午 12 点保存日期,而是在上午 6 点保存日期,即服务器工作时间之前。即使扣除一个小时,也不影响dob。

Calendar now = Calendar.getInstance();
now.setTime(YOUR_DATE);
now.set(Calendar.HOUR_OF_DAY, 6);
YOUR_DATE = now.getTime();

迈克斯的回答很好。但是,如果我们使用 12 pm 保存,如果用户在 12 pm 之前的营业时间内输入日期,则日期比较可能会出错。

于 2014-03-08T16:37:54.537 回答