dbVisitor 6.7.0 解读:公元前日期处理的两种方案

公元前日期在数据库中的处理是一个被长期忽视的难题。Java 的 ISO 8601 年份表示法与数据库的日期系统存在根本性差异,导致跨系统传递公元前日期时经常出现"偏移一年"或"偏移一天"的诡异 Bug。

dbVisitor 6.7.0 新增了 JulianDayTypeHandlerPgDateTypeHandler 两个处理器,分别从"跨库通用"和"PostgreSQL 原生"两个角度彻底解决这个问题。

问题根源:年份表示法的歧义

Java 的 LocalDate 使用 ISO 8601 标准,Year 0 表示公元前 1 年:

Java Year 含义 PostgreSQL 表示
1 公元 1 年 (1 AD) 0001-01-01
0 公元前 1 年 (1 BC) 0001-01-01 BC
-1 公元前 2 年 (2 BC) 0002-01-01 BC
-99 公元前 100 年 (100 BC) 0100-01-01 BC

转换公式:BC 年份 = |Java Year| + 1

java.sql.Date 底层使用 Proleptic Gregorian Calendar,这会在日期转换时引发偏移。不同 JDBC 驱动对公元前日期的处理也各不相同,有些甚至直接抛异常。

方案一:JulianDayTypeHandler --- 跨数据库通用方案

儒略日数(Julian Day Number)是天文学中使用的连续日期计数系统,从公元前 4713 年 1 月 1 日开始,不存在任何历法歧义。

原理 :将 LocalDate 转换为一个 BIGINT 整数存入数据库,读取时逆向还原。

java 复制代码
// 存储:公元前 100 年 → 儒略日数 1684534
LocalDate bcDate = LocalDate.of(-99, 1, 1);

Map<String, Object> params = new HashMap<>();
params.put("id", 1);
params.put("date", bcDate);

jdbcTemplate.executeUpdate(
    "INSERT INTO events (id, julian_day) VALUES (#{id}, #{date, typeHandler=net.hasor.dbvisitor.types.handler.time.JulianDayTypeHandler})",
    params
);

// 读取:儒略日数 1684534 → 公元前 100 年
LocalDate loaded = jdbcTemplate.queryForObject(
    "SELECT julian_day FROM events WHERE id = ?",
    new Object[] { 1 },
    (rs, rowNum) -> new JulianDayTypeHandler().getResult(rs, "julian_day")
);

assertEquals(bcDate, loaded);  // ✔ 通过
assertEquals(-99, loaded.getYear());  // ✔ Year -99 = 100 BC

算法核心(Richards 2012):

java 复制代码
// LocalDate → Julian Day Number
int a = (14 - month) / 12;
int y2 = year + 4800 - a;
int m2 = month + 12 * a - 3;
long jdn = day + (153 * m2 + 2) / 5 + 365 * y2 + y2 / 4 - y2 / 100 + y2 / 400 - 32045;

适用场景

  • 需要跨数据库(MySQL、PostgreSQL、Oracle、SQLite 等)保持一致性
  • 历史学、天文学数据
  • 数据库类型仅需 BIGINT,不依赖原生 DATE

方案二:PgDateTypeHandler --- PostgreSQL 原生方案

如果你的项目锁定 PostgreSQL,可以利用其原生的 BC 后缀格式,直接使用 DATE 类型存储。

java 复制代码
LocalDate bcDate = LocalDate.of(-99, 1, 1);

Map<String, Object> params = new HashMap<>();
params.put("id", 1);
params.put("date", bcDate);

jdbcTemplate.executeUpdate(
    "INSERT INTO events (id, event_date) VALUES (#{id}, #{date, typeHandler=net.hasor.dbvisitor.types.handler.time.PgDateTypeHandler})",
    params
);

// 数据库中存储为: 0100-01-01 BC
// 读取时自动转换回 LocalDate.of(-99, 1, 1)

优势

  • 使用数据库原生 DATE 类型,支持 SQL 中直接查询和比较(如 WHERE event_date < '0500-01-01 BC'
  • 无需额外的类型转换层

注意事项

  • ISO 8601 的闰年规则与 PostgreSQL BC 的闰年规则不同。Java 中 Year -4 是闰年(ISO 闰年),但转换后 5 BC 在 PostgreSQL 中不是闰年。
  • 仅适用于 PostgreSQL

方案对比

维度 JulianDayTypeHandler PgDateTypeHandler
数据库支持 所有(存为 BIGINT) 仅 PostgreSQL
存储类型 BIGINT DATE
SQL 中日期比较 数值比较(可行但不直观) 原生日期比较
精度 天级(无时间) 天级(无时间)
闰年兼容 无歧义(纯数值) 需注意 BC 闰年差异
迁移成本 低(通用整数列) 中(依赖 PG)

选择建议 :跨库项目或对一致性要求高的场景用 JulianDayTypeHandler;PostgreSQL 专属项目且需要在 SQL 中操作日期的场景用 PgDateTypeHandler

相关推荐
Boop_wu9 小时前
[Spring Cloud] 快速上手nacos
后端·spring·spring cloud
浮芷.9 小时前
鸿蒙 6.1 新特性-60fps流畅人物跳跃功能算法深度解析-鸿蒙PC端正弦值计算法
算法·华为·harmonyos·鸿蒙·鸿蒙系统
AI科技星9 小时前
数术工坊·第八卷 大道归一录・番外・下篇 零界封神・万法归元终章
网络·人工智能·算法·几何学·拓扑学
zandy10119 小时前
跨源查询 30 倍提速:衡石 BI 多源异构数据关联技术深度解析
架构·异构
糖果店的幽灵9 小时前
软件测试接口测试从入门到精通:RESTful API设计规范
软件测试·后端·接口测试·restful·设计规范·api设计
Misnearch9 小时前
Leetcode热题100
算法·leetcode·职场和发展
星辰_mya10 小时前
限流、漏斗桶和令牌桶的区别
java·开发语言·面试·架构·高并发
一个被程序员耽误的厨师10 小时前
02-架构篇-前端怎么反客为主把AI编排权拿回到自己手里
前端·人工智能·架构
暗黑小白10 小时前
第六篇:本地模型选型 —— 4 个模型 × 2 种设备 × 2 项任务的全量对比
架构·ai agent
云浪10 小时前
别再让用户干等了:用 Express + SSE 实现《红楼梦》AI 问答实时输出
javascript·后端·node.js