56

我有一个XSD,需要我使用 BigDecimal 来表示纬度/经度。我目前将 lat/lon 作为双精度数,并将它们转换为 BigDecimal,但我只需要使用大约 12 个精度位。我一直无法弄清楚如何设置它。谁能帮我这个?

4

4 回答 4

98

您可以使用setScale()例如

double d = ...
BigDecimal db = new BigDecimal(d).setScale(12, BigDecimal.ROUND_HALF_UP);
于 2012-02-28T13:40:43.840 回答
45

问题的标题询问精度。BigDecimal区分比例和精度。比例是小数位数。您可以将精度视为有效数字的数量,也称为有效数字。

Clojure 中的一些示例。

(.scale     0.00123M) ; 5
(.precision 0.00123M) ; 3

(在 Clojure 中,TheM指定一个 BigDecimal 文字。如果您愿意,可以将 Clojure 翻译成 Java,但我发现它比 Java 更紧凑!)

您可以轻松地增加规模:

(.setScale 0.00123M 7) ; 0.0012300M

但是您不能以完全相同的方式减小比例:

(.setScale 0.00123M 3) ; ArithmeticException Rounding necessary

您还需要通过舍入模式:

(.setScale 0.00123M 3 BigDecimal/ROUND_HALF_EVEN) ;
; Note: BigDecimal would prefer that you use the MathContext rounding
; constants, but I don't have them at my fingertips right now.

所以,很容易改变比例。但是精度呢?这并不像您希望的那么容易!

很容易降低精度:

(.round 3.14159M (java.math.MathContext. 3)) ; 3.14M

但是如何提高精度并不明显:

(.round 3.14159M (java.math.MathContext. 7)) ; 3.14159M (unexpected)

对于持怀疑态度的人,这不仅仅是不显示尾随零的问题:

(.precision (.round 3.14159M (java.math.MathContext. 7))) ; 6 
; (same as above, still unexpected)

FWIW,Clojure 对尾随零很小心,并将显示它们:

4.0000M ; 4.0000M
(.precision 4.0000M) ; 5

回到正轨...您可以尝试使用BigDecimal构造函数,但它不会将精度设置为高于您指定的位数:

(BigDecimal. "3" (java.math.MathContext. 5)) ; 3M
(BigDecimal. "3.1" (java.math.MathContext. 5)) ; 3.1M

所以,没有快速的方法来改变精度。在写这个问题和我正在做的一个项目时,我花了很多时间来解决这个问题。我认为这充其量是一个 CRAZYTOWN API,最坏的情况是一个错误。人们。严重地?

所以,我可以说,如果你想改变精度,你需要执行以下步骤:

  1. 查找当前精度。
  2. 查找当前比例。
  3. 计算尺度变化。
  4. 设置新比例

这些步骤,作为 Clojure 代码:

(def x 0.000691M) ; the input number
(def p' 1) ; desired precision
(def s' (+ (.scale x) p' (- (.precision x)))) ; desired new scale
(.setScale x s' BigDecimal/ROUND_HALF_EVEN)
; 0.0007M

我知道,这有很多步骤只是为了改变精度!

为什么 BigDecimal 还没有提供这个?我忽略了什么吗?

于 2013-12-04T01:43:08.020 回答
4
 BigDecimal decPrec = (BigDecimal)yo.get("Avg");
 decPrec = decPrec.setScale(5, RoundingMode.CEILING);
 String value= String.valueOf(decPrec);

这样,您可以设置 a 的特定精度BigDecimal

decPrec 的值1.5726903423607562595809913132345426 四舍五入为1.57267.

于 2014-01-23T12:20:02.583 回答
0

试试这段代码...

    Integer perc = 5;
    BigDecimal spread = BigDecimal.ZERO; 
    BigDecimal perc = spread.setScale(perc,BigDecimal.ROUND_HALF_UP);
    System.out.println(perc);

结果:0.00000

于 2016-07-11T12:02:24.417 回答