国产av日韩一区二区三区精品,成人性爱视频在线观看,国产,欧美,日韩,一区,www.成色av久久成人,2222eeee成人天堂

目錄
要點(diǎn)總結(jié)
BCMath 的問(wèn)題
案例 3
BCMath 最終指導(dǎo)原則
MySQL 和固定點(diǎn)數(shù)字
PHP 和 MySQL 中的精確數(shù)學(xué)運(yùn)算
結(jié)論
首頁(yè) 后端開(kāi)發(fā) php教程 BCMATH,PHP中的固定點(diǎn)數(shù)學(xué),精確損失案例

BCMATH,PHP中的固定點(diǎn)數(shù)學(xué),精確損失案例

Feb 20, 2025 am 09:17 AM

Fixed Point Math in PHP with BCMath, precision loss cases

PHP 和 MySQL 固定點(diǎn)數(shù)值運(yùn)算的挑戰(zhàn)與技巧

處理固定點(diǎn)數(shù)值時(shí),需要格外小心,尤其是在使用 PHP 和 MySQL 進(jìn)行開(kāi)發(fā)時(shí)。本文將探討使用 PHP BCMath 擴(kuò)展、MySQL 固定點(diǎn)表達(dá)式處理以及將固定點(diǎn)數(shù)據(jù)從 PHP 持久化到 MySQL 時(shí)遇到的障礙和細(xì)節(jié)問(wèn)題。盡管存在一些挑戰(zhàn),我們將嘗試找出如何處理固定點(diǎn)數(shù)值并避免精度損失的方法。

要點(diǎn)總結(jié)

  • PHP 中的 BCMath 擴(kuò)展支持任意精度數(shù)學(xué)運(yùn)算,但如果向其函數(shù)傳遞數(shù)值型變量,則可能導(dǎo)致精度損失。應(yīng)改用表示數(shù)字的字符串值以避免此問(wèn)題。
  • MySQL 支持固定點(diǎn)數(shù)字表達(dá)式,但是,如果任何操作數(shù)采用指數(shù)格式或字符串格式,則會(huì)被視為浮點(diǎn)數(shù)。PHP PDO 擴(kuò)展沒(méi)有用于綁定的 Decimal 參數(shù)類(lèi)型,這可能導(dǎo)致結(jié)果不精確。
  • 為了在 PHP MySQL 應(yīng)用程序中執(zhí)行精確的數(shù)學(xué)運(yùn)算,所有運(yùn)算都可以在 PHP 中處理,并且只有使用 INSERT 或 UPDATE 語(yǔ)句才能將數(shù)據(jù)持久化到 MySQL?;蛘?,可以手動(dòng)構(gòu)建 SQL 查詢,確保所有 SQL 數(shù)學(xué)表達(dá)式都以十進(jìn)制數(shù)表示。

BCMath 的問(wèn)題

BCMath 文檔指出:

為了進(jìn)行任意精度數(shù)學(xué)運(yùn)算,PHP 提供了二進(jìn)制計(jì)算器,它支持任何大小和精度的數(shù)字,這些數(shù)字以字符串表示。

因此,BCMath 函數(shù)參數(shù)應(yīng)以字符串表示。將數(shù)值型變量傳遞給 bcmath 函數(shù)可能導(dǎo)致錯(cuò)誤的結(jié)果,與將雙精度值視為字符串時(shí)出現(xiàn)的精度損失相同。

案例 1

echo bcmul(776.210000, '100', 10) . PHP_EOL;
echo bcmul(776.211000, '100', 10) . PHP_EOL;
echo bcmul(776.210100, '100', 10) . PHP_EOL;

echo bcmul(50018850776.210000, '100', 10) . PHP_EOL;
echo bcmul(50018850776.211000, '100', 10) . PHP_EOL;
echo bcmul(50018850776.210100, '100', 10) . PHP_EOL;

結(jié)果是:

<code>77621.00
77621.100
77621.0100
5001885077621.00
5001885077621.100
5001885077621.00 //此處可見(jiàn)精度損失</code>

切勿將數(shù)值型變量傳遞給 BCMath 函數(shù),只能傳遞表示數(shù)字的字符串值。即使不處理浮點(diǎn)數(shù),BCMath 也會(huì)輸出奇怪的結(jié)果:

案例 2

echo bcmul('10', 0.0001, 10) . PHP_EOL;
echo bcmul('10', 0.00001, 10) . PHP_EOL;
echo 10*0.00001 . PHP_EOL;

結(jié)果是:

<code>0.0010
0 // 這真的很奇怪?。?!
0.0001</code>

其原因是 BCMath 將其參數(shù)轉(zhuǎn)換為字符串,并且在某些情況下,數(shù)字的字符串表示形式具有指數(shù)表示法。

案例 3

echo bcmul('10', '1e-4', 10) . PHP_EOL; // 也輸出 0

PHP 是一種弱類(lèi)型語(yǔ)言,在某些情況下,無(wú)法嚴(yán)格控制輸入——希望處理盡可能多的請(qǐng)求。

例如,我們可以通過(guò)應(yīng)用 sprintf 轉(zhuǎn)換來(lái)“修復(fù)”案例 2案例 3

$val = sprintf("%.10f", '1e-5');
echo bcmul('10', $val, 10) . PHP_EOL;
// 給我們 0.0001000000

但是,應(yīng)用相同的轉(zhuǎn)換會(huì)破壞案例 1 的“正確”行為:

$val = sprintf("%.10f", '50018850776.2100000000');
echo bcmul('10', $val, 10) . PHP_EOL;
echo bcmul('10', 50018850776.2100000000, 10) . PHP_EOL;
500188507762.0999908450 // 錯(cuò)誤
500188507762.10 // 正確

因此,sprintf 解決方案不適用于 BCmath。假設(shè)所有用戶輸入都是字符串,我們可以實(shí)現(xiàn)一個(gè)簡(jiǎn)單的驗(yàn)證器,捕獲所有指數(shù)表示法的數(shù)字并正確轉(zhuǎn)換它們。此技術(shù)在 php-bignumbers 中已實(shí)現(xiàn),因此我們可以安全地傳入類(lèi)似 1e-20 和 50018850776.2101 的參數(shù),而不會(huì)丟失精度。

BCMath 最終指導(dǎo)原則

切勿在 BCMath PHP 擴(kuò)展函數(shù)中將浮點(diǎn)數(shù)用作固定點(diǎn)運(yùn)算參數(shù)。僅使用字符串。

使用 BCMath 擴(kuò)展運(yùn)算時(shí),請(qǐng)注意指數(shù)表示法的參數(shù)。BCMath 函數(shù)無(wú)法正確處理指數(shù)參數(shù)(例如“1e-8”),因此應(yīng)手動(dòng)轉(zhuǎn)換它們。注意,不要使用 sprintf 或類(lèi)似的轉(zhuǎn)換技術(shù),因?yàn)檫@會(huì)導(dǎo)致精度損失。

可以使用 php-bignumbers 庫(kù),它可以處理指數(shù)形式的輸入?yún)?shù),并為用戶提供固定點(diǎn)數(shù)學(xué)運(yùn)算函數(shù)。但是,其性能不如 BCMath 擴(kuò)展,因此它是在健壯的包和性能之間的一種折衷方案。

MySQL 和固定點(diǎn)數(shù)字

在 MySQL 中,固定點(diǎn)數(shù)字使用 DECIMAL 列類(lèi)型處理。您可以閱讀 MySQL 官方文檔了解數(shù)據(jù)類(lèi)型和精確數(shù)學(xué)運(yùn)算。

最有趣的部分是 MySQL 如何處理表達(dá)式:

數(shù)值表達(dá)式的處理取決于表達(dá)式包含的值的類(lèi)型:

如果存在任何近似值,則表達(dá)式為近似值,并使用浮點(diǎn)運(yùn)算進(jìn)行計(jì)算。

如果不存在近似值,則表達(dá)式僅包含精確值。如果任何精確值包含小數(shù)部分(小數(shù)點(diǎn)后的值),則使用 DECIMAL 精確算術(shù)計(jì)算表達(dá)式,精度為 65 位數(shù)字?!熬_”一詞受限于可以在二進(jìn)制中表示的內(nèi)容。例如,1.0/3.0 可以用十進(jìn)制表示法近似為 .333…,但不能寫(xiě)成精確數(shù)字,因此 (1.0/3.0)*3.0 不精確地計(jì)算為 1.0。

否則,表達(dá)式僅包含整數(shù)值。表達(dá)式是精確的,并使用整數(shù)算術(shù)進(jìn)行計(jì)算,精度與 BIGINT 相同(64 位)。

如果數(shù)值表達(dá)式包含任何字符串,則將其轉(zhuǎn)換為雙精度浮點(diǎn)值,表達(dá)式為近似值。

這是一個(gè)簡(jiǎn)短的示例,演示了小數(shù)部分的情況:

echo bcmul(776.210000, '100', 10) . PHP_EOL;
echo bcmul(776.211000, '100', 10) . PHP_EOL;
echo bcmul(776.210100, '100', 10) . PHP_EOL;

echo bcmul(50018850776.210000, '100', 10) . PHP_EOL;
echo bcmul(50018850776.211000, '100', 10) . PHP_EOL;
echo bcmul(50018850776.210100, '100', 10) . PHP_EOL;

這看起來(lái)很簡(jiǎn)單,但讓我們看看如何在 PHP 中處理它。

PHP 和 MySQL 中的精確數(shù)學(xué)運(yùn)算

因此,現(xiàn)在我們必須將固定點(diǎn)值從 PHP 持久化到 MySQL。正確的方法是在查詢中使用預(yù)處理語(yǔ)句和占位符。然后我們進(jìn)行參數(shù)綁定,一切都是安全可靠的。

<code>77621.00
77621.100
77621.0100
5001885077621.00
5001885077621.100
5001885077621.00 //此處可見(jiàn)精度損失</code>

當(dāng)我們將值綁定到語(yǔ)句占位符時(shí),我們可以通過(guò) bindValue 的第三個(gè)參數(shù)指定其類(lèi)型。可能的類(lèi)型由常量 PDO::PARAM_BOOL、PDO::PARAM_NULL、PDO::PARAM_INT、PDO::PARAM_STR、PDO::PARAM_LOB 和 PDO::PARAM_STMT 表示。所以問(wèn)題是 PHP PDO 擴(kuò)展沒(méi)有用于綁定的十進(jìn)制參數(shù)類(lèi)型。結(jié)果,查詢中的所有數(shù)學(xué)表達(dá)式都被視為浮點(diǎn)表達(dá)式,而不是固定點(diǎn)表達(dá)式。

如果我們想利用預(yù)處理語(yǔ)句并使用固定點(diǎn)數(shù)字,最好的方法是在 PHP 中執(zhí)行所有數(shù)學(xué)運(yùn)算并將結(jié)果保存到 MySQL。

echo bcmul('10', 0.0001, 10) . PHP_EOL;
echo bcmul('10', 0.00001, 10) . PHP_EOL;
echo 10*0.00001 . PHP_EOL;

結(jié)論

我們得出以下結(jié)論:

  • 切勿在 BCMath PHP 擴(kuò)展函數(shù)中將浮點(diǎn)數(shù)用作固定點(diǎn)運(yùn)算參數(shù)。僅使用字符串。
  • BCMath 擴(kuò)展不適用于指數(shù)表示法的字符串?dāng)?shù)字。
  • MySQL 支持固定點(diǎn)數(shù)字表達(dá)式,但所有操作數(shù)都必須采用十進(jìn)制格式。如果至少有一個(gè)參數(shù)采用指數(shù)格式或字符串格式,則將其視為浮點(diǎn)數(shù),表達(dá)式將作為浮點(diǎn)數(shù)進(jìn)行計(jì)算。
  • PHP PDO 擴(kuò)展沒(méi)有 Decimal 參數(shù)類(lèi)型,因此,如果您使用預(yù)處理語(yǔ)句并在包含固定點(diǎn)操作數(shù)的 SQL 表達(dá)式中綁定參數(shù),則不會(huì)獲得精確的結(jié)果。
  • 為了在 PHP MySQL 應(yīng)用程序中執(zhí)行精確的數(shù)學(xué)運(yùn)算,您可以選擇兩種方法。第一種方法是在 PHP 中處理所有運(yùn)算,并且只有使用 INSERT 或 UPDATE 語(yǔ)句才能將數(shù)據(jù)持久化到 MySQL。在這種情況下,您可以使用預(yù)處理語(yǔ)句和參數(shù)綁定。第二種方法是手動(dòng)構(gòu)建 SQL 查詢(您仍然可以使用預(yù)處理語(yǔ)句,但必須自行轉(zhuǎn)義參數(shù)),以便所有 SQL 數(shù)學(xué)表達(dá)式都以十進(jìn)制數(shù)表示。

我個(gè)人最喜歡的方法是第一種:在 PHP 中進(jìn)行所有數(shù)學(xué)運(yùn)算。我同意 PHP 和 MySQL 可能不是需要精確數(shù)學(xué)運(yùn)算的應(yīng)用程序的最佳選擇,但是,如果您選擇了這種技術(shù)堆棧,那么了解如何正確處理它是很好的。

(由于篇幅限制,F(xiàn)AQs 部分被省略。如果需要,可以單獨(dú)生成FAQs部分。)

以上是BCMATH,PHP中的固定點(diǎn)數(shù)學(xué),精確損失案例的詳細(xì)內(nèi)容。更多信息請(qǐng)關(guān)注PHP中文網(wǎng)其他相關(guān)文章!

本站聲明
本文內(nèi)容由網(wǎng)友自發(fā)貢獻(xiàn),版權(quán)歸原作者所有,本站不承擔(dān)相應(yīng)法律責(zé)任。如您發(fā)現(xiàn)有涉嫌抄襲侵權(quán)的內(nèi)容,請(qǐng)聯(lián)系admin@php.cn

熱AI工具

Undress AI Tool

Undress AI Tool

免費(fèi)脫衣服圖片

Undresser.AI Undress

Undresser.AI Undress

人工智能驅(qū)動(dòng)的應(yīng)用程序,用于創(chuàng)建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用于從照片中去除衣服的在線人工智能工具。

Clothoff.io

Clothoff.io

AI脫衣機(jī)

Video Face Swap

Video Face Swap

使用我們完全免費(fèi)的人工智能換臉工具輕松在任何視頻中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費(fèi)的代碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

功能強(qiáng)大的PHP集成開(kāi)發(fā)環(huán)境

Dreamweaver CS6

Dreamweaver CS6

視覺(jué)化網(wǎng)頁(yè)開(kāi)發(fā)工具

SublimeText3 Mac版

SublimeText3 Mac版

神級(jí)代碼編輯軟件(SublimeText3)

熱門(mén)話題

如何在PHP中實(shí)施身份驗(yàn)證和授權(quán)? 如何在PHP中實(shí)施身份驗(yàn)證和授權(quán)? Jun 20, 2025 am 01:03 AM

tosecurelyhandleauthenticationandationallizationInphp,lofterTheSesteps:1.AlwaysHashPasswordSwithPassword_hash()andverifyusingspasspassword_verify(),usepreparedStatatementStopreventsqlineptions,andStoreSeruserDatain usseruserDatain $ _sessiveferterlogin.2.implementrole-2.imaccessccsccccccccccccccccccccccccc.

如何在PHP中安全地處理文件上傳? 如何在PHP中安全地處理文件上傳? Jun 19, 2025 am 01:05 AM

要安全處理PHP中的文件上傳,核心在于驗(yàn)證文件類(lèi)型、重命名文件并限制權(quán)限。1.使用finfo_file()檢查真實(shí)MIME類(lèi)型,僅允許特定類(lèi)型如image/jpeg;2.用uniqid()生成隨機(jī)文件名,存儲(chǔ)至非Web根目錄;3.通過(guò)php.ini和HTML表單限制文件大小,設(shè)置目錄權(quán)限為0755;4.使用ClamAV掃描惡意軟件,增強(qiáng)安全性。這些步驟有效防止安全漏洞,確保文件上傳過(guò)程安全可靠。

PHP中==(松散比較)和===(嚴(yán)格的比較)之間有什么區(qū)別? PHP中==(松散比較)和===(嚴(yán)格的比較)之間有什么區(qū)別? Jun 19, 2025 am 01:07 AM

在PHP中,==與===的主要區(qū)別在于類(lèi)型檢查的嚴(yán)格程度。==在比較前會(huì)進(jìn)行類(lèi)型轉(zhuǎn)換,例如5=="5"返回true,而===要求值和類(lèi)型都相同才會(huì)返回true,例如5==="5"返回false。使用場(chǎng)景上,===更安全應(yīng)優(yōu)先使用,==僅在需要類(lèi)型轉(zhuǎn)換時(shí)使用。

如何在PHP( - , *, /,%)中執(zhí)行算術(shù)操作? 如何在PHP( - , *, /,%)中執(zhí)行算術(shù)操作? Jun 19, 2025 pm 05:13 PM

PHP中使用基本數(shù)學(xué)運(yùn)算的方法如下:1.加法用 號(hào),支持整數(shù)和浮點(diǎn)數(shù),也可用于變量,字符串?dāng)?shù)字會(huì)自動(dòng)轉(zhuǎn)換但不推薦依賴;2.減法用-號(hào),變量同理,類(lèi)型轉(zhuǎn)換同樣適用;3.乘法用*號(hào),適用于數(shù)字及類(lèi)似字符串;4.除法用/號(hào),需避免除以零,并注意結(jié)果可能是浮點(diǎn)數(shù);5.取模用%號(hào),可用于判斷奇偶數(shù),處理負(fù)數(shù)時(shí)余數(shù)符號(hào)與被除數(shù)一致。正確使用這些運(yùn)算符的關(guān)鍵在于確保數(shù)據(jù)類(lèi)型清晰并處理好邊界情況。

如何與PHP的NOSQL數(shù)據(jù)庫(kù)(例如MongoDB,Redis)進(jìn)行交互? 如何與PHP的NOSQL數(shù)據(jù)庫(kù)(例如MongoDB,Redis)進(jìn)行交互? Jun 19, 2025 am 01:07 AM

是的,PHP可以通過(guò)特定擴(kuò)展或庫(kù)與MongoDB和Redis等NoSQL數(shù)據(jù)庫(kù)交互。首先,使用MongoDBPHP驅(qū)動(dòng)(通過(guò)PECL或Composer安裝)創(chuàng)建客戶端實(shí)例并操作數(shù)據(jù)庫(kù)及集合,支持插入、查詢、聚合等操作;其次,使用Predis庫(kù)或phpredis擴(kuò)展連接Redis,執(zhí)行鍵值設(shè)置與獲取,推薦phpredis用于高性能場(chǎng)景,Predis則便于快速部署;兩者均適用于生產(chǎn)環(huán)境且文檔完善。

我如何了解最新的PHP開(kāi)發(fā)和最佳實(shí)踐? 我如何了解最新的PHP開(kāi)發(fā)和最佳實(shí)踐? Jun 23, 2025 am 12:56 AM

TostaycurrentwithPHPdevelopmentsandbestpractices,followkeynewssourceslikePHP.netandPHPWeekly,engagewithcommunitiesonforumsandconferences,keeptoolingupdatedandgraduallyadoptnewfeatures,andreadorcontributetoopensourceprojects.First,followreliablesource

什么是PHP,為什么它用于Web開(kāi)發(fā)? 什么是PHP,為什么它用于Web開(kāi)發(fā)? Jun 23, 2025 am 12:55 AM

PHPbecamepopularforwebdevelopmentduetoitseaseoflearning,seamlessintegrationwithHTML,widespreadhostingsupport,andalargeecosystemincludingframeworkslikeLaravelandCMSplatformslikeWordPress.Itexcelsinhandlingformsubmissions,managingusersessions,interacti

如何設(shè)置PHP時(shí)區(qū)? 如何設(shè)置PHP時(shí)區(qū)? Jun 25, 2025 am 01:00 AM

tosetTherightTimeZoneInphp,restate_default_timezone_set()functionAtthestArtofyourscriptWithavalIdidentIdentifiersuchas'america/new_york'.1.usedate_default_default_timezone_set_set()

See all articles