Mysql

关注公众号 jb51net

关闭
首页 > 数据库 > Mysql > MySQL时间类型与Java日期时间类

MySQL时间类型与Java日期时间类对应关系示例详解

作者:比特森林探险记

在开发中数据库和编程语言通常需要处理日期和时间数据,MySQL数据库和Java编程语言都提供了各种时间类型来处理日期和时间数据,这篇文章主要介绍了MySQL时间类型与Java日期时间类对应关系的相关资料,需要的朋友可以参考下

一、MySQL 时间类型与 Java 类型对应关系

MySQL 类型范围/描述Java 对应类型精度级别
​DATE​'1000-01-01' - '9999-12-31'java.time.LocalDate日期
​TIME​'-838:59:59' - '838:59:59'java.time.LocalTime时间
​DATETIME​'1000-01-01 00:00:00' - '9999-12-31 23:59:59'java.time.LocalDateTime毫秒/微秒
​TIMESTAMP​'1970-01-01 00:00:01' - '2038-01-19 03:14:07'java.time.Instantjava.sql.Timestamp秒/毫秒
​YEAR​1901 - 2155java.time.Yearjava.lang.Integer

二、JDBC 操作映射关系

1. 从数据库读取数据

try (ResultSet rs = stmt.executeQuery("SELECT * FROM time_table")) {
    while (rs.next()) {
        LocalDate date = rs.getObject("date_column", LocalDate.class);
        LocalTime time = rs.getObject("time_column", LocalTime.class);
        LocalDateTime dateTime = rs.getObject("datetime_column", LocalDateTime.class);
        Instant timestamp = rs.getObject("ts_column", Instant.class);
    }
}

2. 向数据库写入数据

PreparedStatement ps = conn.prepareStatement(
    "INSERT INTO time_table (date_column, time_column, datetime_column, ts_column) " +
    "VALUES (?, ?, ?, ?)");
    
ps.setObject(1, LocalDate.now());
ps.setObject(2, LocalTime.now());
ps.setObject(3, LocalDateTime.now());
ps.setObject(4, Instant.now());

ps.executeUpdate();

三、MySQL 时间类型详细对比

​特性​DATETIMEDATETIMETIMESTAMPYEAR
​时区敏感​​是​
​自动初始化​​是​
​自动更新​​是​
​存储空间(字节)​335-841
​微秒精度支持​-MySQL 5.6+MySQL 5.6+MySQL 5.6.4+-
​2038年限制​​有​​ (2038年)

四、微秒精度处理(MySQL 5.6+)

// Java 端处理微秒(MySQL DATETIME(6))
LocalDateTime ldt = LocalDateTime.now()
    .withNano(456000000); // 456毫秒

// MySQL 创建表
CREATE TABLE events (
    event_time DATETIME(6) // 支持微秒级精度
);

五、时区处理最佳实践

1. 时间存储策略

// 写入统一使用 UTC 时间
ps.setObject(4, Instant.now()); // UTC 时间戳

// 读取时转换为本地时间
Instant dbTime = rs.getObject("event_time", Instant.class);
ZonedDateTime localTime = dbTime.atZone(ZoneId.systemDefault());

2. MySQL 服务器配置

-- 检查时区设置
SELECT @@global.time_zone, @@session.time_zone;

-- 推荐设置为 UTC
SET GLOBAL time_zone = '+00:00';

六、特殊场景处理方案

1. 日期范围统计

LocalDate start = LocalDate.of(2023, 1, 1);
LocalDate end = LocalDate.of(2023, 12, 31);

PreparedStatement ps = conn.prepareStatement(
    "SELECT * FROM orders WHERE order_date BETWEEN ? AND ?");
    
ps.setObject(1, start);
ps.setObject(2, end);

2. 时间区间查询(TIMESTAMP + 时区)

Instant startInstant = ZonedDateTime.of(
    2023, 5, 1, 0, 0, 0, 0, 
    ZoneId.of("Asia/Shanghai")).toInstant();

PreparedStatement ps = conn.prepareStatement(
    "SELECT * FROM transactions WHERE create_time >= ?");
    
ps.setObject(1, startInstant);

3. 时间间隔计算(Java 端处理)

LocalDateTime orderTime = rs.getObject("order_time", LocalDateTime.class);
LocalDateTime deliveryTime = rs.getObject("delivery_time", LocalDateTime.class);

Duration interval = Duration.between(orderTime, deliveryTime);
System.out.printf("配送耗时: %.1f小时", interval.toMinutes() / 60.0);

七、传统 java.sql 类替代方案

传统类Java 8+ 替代方案使用场景
java.sql.DateLocalDate只处理日期部分
java.sql.TimeLocalTime只处理时间部分
java.sql.TimestampInstant 或 LocalDateTime带时区的时间戳/本地日期时间

转换示例:

// 传统转现代
Timestamp ts = rs.getTimestamp("create_time");
LocalDateTime ldt = ts.toLocalDateTime();

// 现代转传统
LocalDateTime now = LocalDateTime.now();
Timestamp ts = Timestamp.valueOf(now);

八、时区转换流程图

graph LR
    A[业务系统时间\nLocalDateTime] -->|应用时区| B[JVM 默认时区]
    B -->|转换为UTC| C[Instant/UTC时间]
    C -->|存储至MySQL| D[TIMESTAMP列]
    D -->|读取为Instant| E[Java Instant类型]
    E -->|转换至目标时区| F[用户时区显示]

九、常见问题解决方案

总结建议

掌握MySQL和Java时间类型的精确对应关系,是处理国际化应用、财务系统和时效敏感业务的基础。遵循UTC存储、本地化展示的原则,可规避90%的时区相关问题。

到此这篇关于MySQL时间类型与Java日期时间类对应关系的文章就介绍到这了,更多相关MySQL时间类型与Java日期时间类内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文