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

目錄
要點總結
BCMath 的問題
案例 3
BCMath 最終指導原則
MySQL 和固定點數(shù)字
PHP 和 MySQL 中的精確數(shù)學運算
結論
首頁 後端開發(fā) php教程 BCMATH,PHP中的固定點數(shù)學,精確損失案例

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

Feb 20, 2025 am 09:17 AM

Fixed Point Math in PHP with BCMath, precision loss cases

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

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

要點總結

  • PHP 中的 BCMath 擴展支持任意精度數(shù)學運算,但如果向其函數(shù)傳遞數(shù)值型變量,則可能導致精度損失。應改用表示數(shù)字的字符串值以避免此問題。
  • MySQL 支持固定點數(shù)字表達式,但是,如果任何操作數(shù)採用指數(shù)格式或字符串格式,則會被視為浮點數(shù)。 PHP PDO 擴展沒有用於綁定的 Decimal 參數(shù)類型,這可能導致結果不精確。
  • 為了在 PHP MySQL 應用程序中執(zhí)行精確的數(shù)學運算,所有運算都可以在 PHP 中處理,並且只有使用 INSERT 或 UPDATE 語句才能將數(shù)據(jù)持久化到 MySQL。或者,可以手動構建 SQL 查詢,確保所有 SQL 數(shù)學表達式都以十進制數(shù)表示。

BCMath 的問題

BCMath 文檔指出:

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

因此,BCMath 函數(shù)參數(shù)應以字符串表示。將數(shù)值型變量傳遞給 bcmath 函數(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;

結果是:

<code>77621.00
77621.100
77621.0100
5001885077621.00
5001885077621.100
5001885077621.00 //此處可見精度損失</code>

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

案例 2

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

結果是:

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

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

案例 3

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

PHP 是一種弱類型語言,在某些情況下,無法嚴格控制輸入——希望處理盡可能多的請求。

例如,我們可以通過應用 sprintf 轉(zhuǎn)換來“修復”案例 2案例 3

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

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

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

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

BCMath 最終指導原則

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

使用 BCMath 擴展運算時,請注意指數(shù)表示法的參數(shù)。 BCMath 函數(shù)無法正確處理指數(shù)參數(shù)(例如“1e-8”),因此應手動轉(zhuǎn)換它們。注意,不要使用 sprintf 或類似的轉(zhuǎn)換技術,因為這會導致精度損失。

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

MySQL 和固定點數(shù)字

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

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

數(shù)值表達式的處理取決於表達式包含的值的類型:

如果存在任何近似值,則表達式為近似值,並使用浮點運算進行計算。

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

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

如果數(shù)值表達式包含任何字符串,則將其轉(zhuǎ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;

這看起來很簡單,但讓我們看看如何在 PHP 中處理它。

PHP 和 MySQL 中的精確數(shù)學運算

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

<code>77621.00
77621.100
77621.0100
5001885077621.00
5001885077621.100
5001885077621.00 //此處可見精度損失</code>

當我們將值綁定到語句佔位符時,我們可以通過 bindValue 的第三個參數(shù)指定其類型??赡艿念愋陀沙A?PDO::PARAM_BOOL、PDO::PARAM_NULL、PDO::PARAM_INT、PDO::PARAM_STR、PDO::PARAM_LOB 和 PDO::PARAM_STMT 表示。所以問題是 PHP PDO 擴展沒有用於綁定的十進制參數(shù)類型。結果,查詢中的所有數(shù)學表達式都被視為浮點表達式,而不是固定點表達式。

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

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

結論

我們得出以下結論:

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

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

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

以上是BCMATH,PHP中的固定點數(shù)學,精確損失案例的詳細內(nèi)容。更多資訊請關注PHP中文網(wǎng)其他相關文章!

本網(wǎng)站聲明
本文內(nèi)容由網(wǎng)友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發(fā)現(xiàn)涉嫌抄襲或侵權的內(nèi)容,請聯(lián)絡admin@php.cn

熱AI工具

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅(qū)動的應用程序,用於創(chuàng)建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發(fā)環(huán)境

Dreamweaver CS6

Dreamweaver CS6

視覺化網(wǎng)頁開發(fā)工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

如何在PHP中實施身份驗證和授權? 如何在PHP中實施身份驗證和授權? 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中的文件上傳,核心在於驗證文件類型、重命名文件並限制權限。 1.使用finfo_file()檢查真實MIME類型,僅允許特定類型如image/jpeg;2.用uniqid()生成隨機文件名,存儲至非Web根目錄;3.通過php.ini和HTML表單限製文件大小,設置目錄權限為0755;4.使用ClamAV掃描惡意軟件,增強安全性。這些步驟有效防止安全漏洞,確保文件上傳過程安全可靠。

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

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

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

PHP中使用基本數(shù)學運算的方法如下:1.加法用 號,支持整數(shù)和浮點數(shù),也可用於變量,字符串數(shù)字會自動轉(zhuǎn)換但不推薦依賴;2.減法用-號,變量同理,類型轉(zhuǎn)換同樣適用;3.乘法用*號,適用於數(shù)字及類似字符串;4.除法用/號,需避免除以零,並註意結果可能是浮點數(shù);5.取模用%號,可用於判斷奇偶數(shù),處理負數(shù)時餘數(shù)符號與被除數(shù)一致。正確使用這些運算符的關鍵在於確保數(shù)據(jù)類型清晰並處理好邊界情況。

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

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

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

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

什麼是PHP,為什麼它用於Web開發(fā)? 什麼是PHP,為什麼它用於Web開發(fā)? Jun 23, 2025 am 12:55 AM

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

如何設置PHP時區(qū)? 如何設置PHP時區(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