【问题标题】:Data type for nanosecond time precision in JavaJava中纳秒时间精度的数据类型
【发布时间】:2021-03-13 05:29:31
【问题描述】:

我想要一个包含纳秒精度整数的数值变量。我试过这个:

Instant t = Instant.now();
a = BigDecimal.valueof(t.getEpochSecond() * 1e9 + t.getNano());
b = BigDecimal.valueof(t.getEpochSecond() * 1e9 + t.getNano() + 1);

ab 都包含相同的值(例如 1.60681511777265408E+18),+1 操作在这里丢失。

知道如何处理这种精度吗?它的目标是为 InfluxDB 中的时间列保留纳秒精度的时间戳。我知道它与双精度有关 (Java BigDecimal difference),但我还没有找到任何可行的解决方案。

【问题讨论】:

  • 一个问题是 1e9 是一个double。这意味着valueOf 的参数表达式是以浮点数计算的......并且您正在失去精度。
  • 另外值得注意的是,纳秒时钟可能没有纳秒精度。事实上,答案中显示的输出暗示您可能会得到微秒精度。
  • 为什么要使用数字类型? Instant 已经具有纳秒精度。它不是给你想要的一切吗?
  • @OleV.V. Instant 每秒保存纳秒,但不能给你一个值,我需要在 InfluxDB 中为值加上时间戳。
  • 感谢您的回答。说得通。如果是我,如果 InfluxDB 有 timestamp with time zone 类型,我会使用它,否则将 Instant 转换为字符串并存储为 charvarchar 以提高可读性。

标签: java time types type-conversion time-precision


【解决方案1】:

只需使用BigDecimal中的方法进行操作即可。并确保您不使用valueOf(double)。不过,您可以使用valueOf(long)

BigDecimal a = BigDecimal.valueOf(t.getEpochSecond())
            .multiply(BigDecimal.valueOf(1_000_000_000))
            .add(BigDecimal.valueOf(t.getNano()));

BigDecimal b = BigDecimal.valueOf(t.getEpochSecond())
            .multiply(BigDecimal.valueOf(1_000_000_000))
            .add(BigDecimal.valueOf(t.getNano()))
            .add(BigDecimal.ONE);

打印 ab 会得到类似的结果:

1606815981554921000
1606815981554921001

【讨论】:

    【解决方案2】:

    如果您要存储一个整数值,为什么要使用 BigDecimal 而不是 BigInteger

    import java.time.*;
    import java.math.*;
    
    public class MyClass {
        public static void main(String args[]) {
          Instant t = Instant.now();
          BigInteger a = BigInteger.valueOf(t.getEpochSecond() * 1_000_000_000
                  + t.getNano());
          BigInteger b = BigInteger.valueOf(t.getEpochSecond() * 1_000_000_000
                  + t.getNano() + 1);
    
          System.out.println(a); // 1606816120696314000
          System.out.println(b); // 1606816120696314001
        }
    }
    

    但需要在 2262 年切换到 BigInteger 数学方法,因为 long 参数将开始溢出。

    【讨论】:

    • 感谢您的提示。
    猜你喜欢
    • 2015-02-27
    • 1970-01-01
    • 2017-04-22
    • 2023-03-17
    • 1970-01-01
    • 1970-01-01
    • 2015-07-20
    • 2023-04-07
    • 2023-02-10
    相关资源
    最近更新 更多