如何在 Java 中执行 Golang 等于 BigDecimal.devide(BigDecimal divisor, MathContext mc) 的精确除法

How to perform precission deviding in Golang equal to BigDecimal.devide(BigDecimal divisor, MathContext mc) in java

提问人:rizalPurnama 提问时间:11/2/2022 更新时间:11/6/2022 访问量:405

问:

我的任务是将核心银行系统从 java 迁移到 golang,但是当我尝试在 golang 中执行精确分隔时,我遇到了困难。我尝试过使用数学/包舍入函数,但我没有得到我想要的结果。我认为问题是我在 java 中找不到等于 BigDecimal.devide(BigDecimal divisor, MathContext mc) 的 golang 除法函数。

谁能帮我得到我想要的结果,请..

非常感谢..

这是我的 java 代码,它是期望的结果


public BigDecimal getGrossInterest(BigDecimal a, BigDecimal b, Integer c) {
    BigDecimal d = a.multiply(b).multiply(new BigDecimal(c));
    BigDecimal e = d.divide(new BigDecimal(365), MathContext.DECIMAL32);
    BigDecimal result = e.setScale(0, RoundingMode.HALF_UP);
    return result;
}

@Test
void TestGetGrossInterest() {
    BigDecimal a1 = new BigDecimal(100000000);
    BigDecimal b1 = new BigDecimal(0.045);
    Integer c1 = Integer.valueOf(28);

    BigDecimal result1 = getGrossInterest(a1, b1, c1);
    System.out.println(result1); // result1 is 345206

    BigDecimal a2 = new BigDecimal(10000000);
    BigDecimal b2 = new BigDecimal(0.035);
    Integer c2 = Integer.valueOf(16);

    BigDecimal result2 = getGrossInterest(a2, b2, c2);
    System.out.println(result2); // result2 is 15342
}

和我的Golang代码,结果错误,如下所示。


func getGrossInterest(a float64, b float64, c int) float64 {
    d := a * b * float64(c)
    e := d / 365
    return math.Round(e)
}

func TestGetGrossInterest(t *testing.T) {
    a1 := 100000000.00
    b1 := 0.045
    c1 := 28

    result1 := getGrossInterest(a1, b1, c1)
    fmt.Printf("result1: %f\n", result1) 
    assert.Equal(t, float64(345206), result1) // failed test because result1 is 345205

    a2 := 10000000.00
    b2 := 0.035
    c2 := 16

    result2 := getGrossInterest(a2,b2,c2)
    fmt.Printf("result2: %f\n", result2)
    assert.Equal(t, float64(15342), result2)
}

Java Go 浮点 精度 小数点

评论

2赞 0x4e696b68696c 11/2/2022
Float 在 Golang 或大多数语言中给出精确的结果,您应该使用十进制包来进行业务计算

答:

0赞 2r2w 11/2/2022 #1

我建议使用 github.com/shopspring/decimal 包并使用

https://pkg.go.dev/github.com/shopspring/decimal#Decimal.DivRound 方法

enter image description here

0赞 rizalPurnama 11/3/2022 #2

@2r2w感谢您的回答。

我试过了,但我没有得到我想要的结果。func (Decimal) DivRound

这是我的代码:

爪哇岛

@Test
void test() {
    BigDecimal a = BigDecimal.valueOf(125999999.99999999533706329657434253022074699401855468750000000000);
    BigDecimal b = BigDecimal.valueOf(5600000.0000000005329070518200751394033432006835937500000000000);

    BigDecimal resultA = a.divide(new BigDecimal(365), MathContext.DECIMAL32);
    System.out.println(resultA); // resultA is 345205.5

    BigDecimal resultB = b.divide(new BigDecimal(365), MathContext.DECIMAL32);
    System.out.println(resultB); // resultB is 15342.47
}

戈朗

func TestXxx(t *testing.T) {
    a := decimal.NewFromFloat(125999999.99999999533706329657434253022074699401855468750000000000)
    b := decimal.NewFromFloat(5600000.0000000005329070518200751394033432006835937500000000000)

    resultAWithPrec1 := a.DivRound(decimal.NewFromInt(365), 1)
    fmt.Printf("%v\n", resultAWithPrec1.String()) // result is 345205.5 (Correct)

    resultBWithPrec1 := b.DivRound(decimal.NewFromInt(365), 1)
    fmt.Printf("%v\n", resultBWithPrec1.String()) // result is 15342.5 (Incorrect)

    resultAWithPrec2 := a.DivRound(decimal.NewFromInt(365), 2)
    fmt.Printf("%v\n", resultAWithPrec2.String()) // result is 345205.48 (Incorrect)

    resultBWithPrec2 := b.DivRound(decimal.NewFromInt(365), 2)
    fmt.Printf("%v\n", resultBWithPrec2.String()) // result is 15342.47 (Correct)
}

我认为问题是:

Java 可以生成不同的十进制数(345205.5 和 15342.47),而在 Golang 上,我必须定义特定的十进制数。

如何达到相同的结果?