在 C++ 和 Oracle 数据库之间传递数据时的浮点精度问题

Posted

技术标签:

【中文标题】在 C++ 和 Oracle 数据库之间传递数据时的浮点精度问题【英文标题】:Floating point precision issue when passing data between C++ and Oracle database 【发布时间】:2013-06-11 01:08:28 【问题描述】:

我们有一个设置,其中远程 C++ 客户端调用 Oracle 函数(使用 OCCI 库)传递一些数据。然后,此 Oracle 函数将数据插入到表中。传递给函数的数据之一是 AMOUNT,它是一个浮点数。

在 C++ 客户端中,此 AMOUNT 变量定义为“双精度”。 Oracle 函数和表都将此列的数据类型定义为“NUMBER”(没有明确定义任何精度或小数位数)。

我们发现客户端发送的内容与我们在数据库中接收和存储的内容之间存在很多不一致。例如,客户声称他们传递了一个值“35.6”,但我们在表中看到的是“35.59999847”。

我们应该以最多 6 位小数的精度存储这些值。如果我将 AMOUNT 列的数据类型定义从 NUMBER 更改为 NUMBER(38,6),我得到“35.599998”。如果我将其更改为 NUMBER (38,5),我最终会得到“35.6”。

有人可以建议这里可能发生的事情吗?我知道不建议将金额或价格值视为浮点数,而应使用整数,但在这种情况下,我们无法控制 C++ 客户端。我们在大量数据中发现了这个问题(超过 50% 的数据存在这个问题)。

此外,此问题不仅限于浮点数。即使整数很大,我们也可以看到不一致(例如,作为 1000000000 传递的值被存储为 1000000004)。

【问题讨论】:

floating-point-gui.de Floating point error in representation?的可能重复 如果 C++ 客户端使用 DECIMAL,而数据库使用 NUMBER,那么浮点到底是从哪里来的? @EJP - 抱歉,意思是“双”而不是“十进制”。编辑了我的帖子。 @OliCharlesworth,是的,我以前看过那个网站。但我从没想过这个问题会太广泛。此外,同一个 C++ 客户端也在调用其他系统(例如 mysql 数据库)。他们是同步的。到目前为止,此问题仅在 C++ 和 Oracle 集成中出现。 【参考方案1】:

您可能无法控制客户端,但您可以更改界面吗?

将浮点数传递给 Oracle 将不可避免地导致此问题,因为它们本质上是不精确的数据类型。但是,如果您可以修改接口,您可以将两个整数传递给 Oracle:AMOUNT 的整数和 AMOUNT 的小数部分。然后,Oracle 可以将这两个整数组合成一个 NUMBER 变量,它可以愉快地使用它。


自从我写这篇文章以来,你已经编辑了你的问题,正如 Oli 指出的那样,这个建议不再有效。

你能做些什么呢?

    通过在 Oracle 层强制执行小数精度来伪造它。这可能会产生一些舍入错误,但比现在少得多。虽然它不能解决大整数的问题。 将此作为一个错误向 C++ 客户端的开发人员提出。 (专业提示:在提交错误报告时,尽量不要称他们为愚蠢的人)。 告诉您的用户他们只需要忍受它。

至于为什么这是 Oracle 而不是您的其他系统的问题,MySQL 支持 FLOAT 和 DOUBLE,并且可能其他系统也支持。 Oracle 一直对数据完整性非常严格,浮点数实在是太松懈了。

【讨论】:

但是已经太晚了。 OP表示客户端将值存储在double中,因此精度已经丢失。 @OliCharlesworth - 在我发布此内容后,OP 改变了这一点。虽然我同意这确实会使我的建议无效。

以上是关于在 C++ 和 Oracle 数据库之间传递数据时的浮点精度问题的主要内容,如果未能解决你的问题,请参考以下文章

请建议在 C# 和 C++ 之间传递数据的基础设施

什么类型的数据可以在预编译的 C++ 和 Java for Android 之间传递?

如何在 c++ winform 中的线程之间传递数据?

如何通过 emscripten 在 C++ 和 javascript 之间传递字符串

在C语言中,函数实参与形参之间的数据传递方式是( )传递方式。

MFC的WebBrowser控件 C++与JavaScript之间数据交互传递