Estou desenvolvendo um aplicativo que armazena alguns dados com registro de data e hora no passado.
Minha JVM é IBM Semeru (17) rodando em Europe/Paris tz. Mas eu quero armazenar o timestamp em UTC (GMT+0).
Abaixo está meu código:
<dependencies>
<dependency>
<groupId>org.postgresql</groupId>
<artifactId>postgresql</artifactId>
<version>42.7.1</version>
</dependency>
</dependencies>
public class Main {
/*
create table theentity (
theid integer not null,
thevalue timestamp(6),
primary key (theid)
)
*/
public static void main(String[] args) {
TimeZone.setDefault( TimeZone.getTimeZone( ZoneId.of( "Europe/Paris" ) ) );
LocalDateTime d_1900_01_01_T_00_09_23 = LocalDateTime.of( 1900, 1, 1, 0, 9, 23, 0 );
LocalDateTime d_1900_01_01_T_00_09_22 = LocalDateTime.of( 1900, 1, 1, 0, 9, 22, 0 );
LocalDateTime d_1900_01_01_T_00_09_21 = LocalDateTime.of( 1900, 1, 1, 0, 9, 21, 0 );
LocalDateTime d_1900_01_01_T_00_09_20 = LocalDateTime.of( 1900, 1, 1, 0, 9, 20, 0 );
LocalDateTime d_1900_01_01_T_00_09_19 = LocalDateTime.of( 1900, 1, 1, 0, 9, 19, 0 );
try(Connection c = DriverManager.getConnection( "jdbc:postgresql://localhost:5432/hibernate_orm_test?preparedStatementCacheQueries=0&escapeSyntaxCallMode=callIfNoReturn",
"postgres", "root")) {
PreparedStatement p = c.prepareStatement( "insert into theentity values(?, ?)" );
bindAndExecute( p, 1, d_1900_01_01_T_00_09_23 );
bindAndExecute( p, 2, d_1900_01_01_T_00_09_22 );
bindAndExecute( p, 3, d_1900_01_01_T_00_09_21 );
bindAndExecute( p, 4, d_1900_01_01_T_00_09_20 );
bindAndExecute( p, 5, d_1900_01_01_T_00_09_19 );
} catch (Exception e) {
e.printStackTrace();
}
}
private static void bindAndExecute(PreparedStatement p, int id, LocalDateTime localDateTime)
throws SQLException {
p.setInt( 1, id );
p.setTimestamp(2,
Timestamp.valueOf( localDateTime ),
Calendar.getInstance( TimeZone.getTimeZone( ZoneId.of( "GMT" ) ) )
);
p.executeUpdate();
}
}
Em outras palavras, tentei persistir 5 carimbos de data/hora (observe que esses carimbos de data/hora são baseados na Europa/Paris por causa de TimeZone.setDefault( TimeZone.getTimeZone( ZoneId.of( "Europe/Paris" ) ) );
)
- 1900-01-01T00:09:23
- 1900-01-01T00:09:22
- 1900-01-01T00:09:21
- 1900-01-01T00:09:20
- 1900-01-01T00:09:19
Esta linha traduziria Europa/Paris tz para UTC tz:
p.setTimestamp(2,
Timestamp.valueOf( localDateTime ),
Calendar.getInstance( TimeZone.getTimeZone( ZoneId.of( "GMT" ) ) ) );
Execute-o para criar 5 linhas no postresql:
theid | thevalue
-------+---------------------
1 | 1900-01-01 00:00:02
2 | 1900-01-01 00:00:01
3 | 1900-01-01 00:00:00
4 | 1899-12-31 23:09:20
5 | 1899-12-31 23:09:19
(5 rows)
A maioria de vocês pode pensar que Paris está em GMT+1. Está correto, mas NÃO ESTAVA correto. Às 19h00, estava em GMT+00:09:21 (9 minutos, 21 seg)!!
Então os primeiros 3 timestamps foram salvos corretamente na tabela. Mas, o estranho está na linha 4. Está, 1899-12-31 23:09:20
mas eu esperava que estivesse 1899-12-31 23:59:59
. Mas parece que em 1899, a antiga API DateTime acha que está em GMT+1. Isso fez com que as linhas 4, 5 e assim por diante... estivessem erradas!!
Vocês podem explicar isso?