摘要:上次講到是如何解析大整數(shù)的,一筆帶過了的處理,再詳細(xì)閱讀該函數(shù)的源碼,以下是小分析??偨Y(jié)閱讀完這個函數(shù)的源碼,學(xué)習(xí)到的是浮動數(shù)與字符串的互相轉(zhuǎn)換的實現(xiàn)細(xì)節(jié),字符串與浮點數(shù)之間的關(guān)系較復(fù)雜,之后還要繼續(xù)學(xué)習(xí)。
上次講到PHP是如何解析大整數(shù)的,一筆帶過了number_format的處理,再詳細(xì)閱讀該函數(shù)的源碼,以下是小分析。
函數(shù)原型string number_format ( float $number [, int $decimals = 0 ] ) string number_format ( float $number , int $decimals = 0 , string $dec_point = "." , string $thousands_sep = "," )
函數(shù)可以接受1、2、4個參數(shù)(具體可以看代碼的實現(xiàn))。
如果只提供第一個參數(shù),number的小數(shù)部分會被去掉,并且每個千位分隔符都是英文小寫逗號"," ;
如果提供兩個參數(shù),number將保留小數(shù)點后的位數(shù)到你設(shè)定的值,其余同樓上;
如果提供了四個參數(shù),number 將保留decimals個長度的小數(shù)部分, 小數(shù)點被替換為dec_point,千位分隔符替換為thousands_sep
// number // 你要格式化的數(shù)字 // num_decimal_places // 要保留的小數(shù)位數(shù) // dec_separator // 指定小數(shù)點顯示的字符 // thousands_separator // 指定千位分隔符顯示的字符 /* {{{ proto string number_format(float number [, int num_decimal_places [, string dec_separator, string thousands_separator]]) Formats a number with grouped thousands */ PHP_FUNCTION(number_format) { // 期望number_format的第一個參數(shù)num是double類型的,在詞法階段已經(jīng)對字面量常量做了轉(zhuǎn)換 double num; zend_long dec = 0; char *thousand_sep = NULL, *dec_point = NULL; char thousand_sep_chr = ",", dec_point_chr = "."; size_t thousand_sep_len = 0, dec_point_len = 0; // 解析參數(shù) ZEND_PARSE_PARAMETERS_START(1, 4) Z_PARAM_DOUBLE(num)// 拿到double類型的num Z_PARAM_OPTIONAL Z_PARAM_LONG(dec) Z_PARAM_STRING_EX(dec_point, dec_point_len, 1, 0) Z_PARAM_STRING_EX(thousand_sep, thousand_sep_len, 1, 0) ZEND_PARSE_PARAMETERS_END(); switch(ZEND_NUM_ARGS()) { case 1: RETURN_STR(_php_math_number_format(num, 0, dec_point_chr, thousand_sep_chr)); break; case 2: RETURN_STR(_php_math_number_format(num, (int)dec, dec_point_chr, thousand_sep_chr)); break; case 4: if (dec_point == NULL) { dec_point = &dec_point_chr; dec_point_len = 1; } if (thousand_sep == NULL) { thousand_sep = &thousand_sep_chr; thousand_sep_len = 1; } // _php_math_number_format_ex // 真正處理的函數(shù),在本文件第1107行 RETVAL_STR(_php_math_number_format_ex(num, (int)dec, dec_point, dec_point_len, thousand_sep, thousand_sep_len)); break; default: WRONG_PARAM_COUNT; } } /* }}} */代碼執(zhí)行流程圖 _php_math_number_format_ex
函數(shù)實現(xiàn)的各種參數(shù)數(shù)量,最終都會調(diào)用_php_math_number_format_ex函數(shù)。函數(shù)主要做的是:
strpprintf處理負(fù)數(shù);
根據(jù)要保留的小數(shù)點對浮點數(shù)進(jìn)行四舍五入;
調(diào)用strpprintf函數(shù)將浮點數(shù)表達(dá)式轉(zhuǎn)成字符串表示;
計算需要分配給結(jié)果變量的字符串長度;
將結(jié)果拷貝到返回值中(如果有千位符,則進(jìn)行千位符分割)
這個函數(shù)是實現(xiàn)浮點數(shù)與字符串的轉(zhuǎn)換,如上文所說,最終是調(diào)用了php_conv_fp函數(shù)做的轉(zhuǎn)換(這里是通過gdb調(diào)試做的定位),而php_conv_fp函數(shù),往下追蹤,調(diào)用的是zend_dtoa函數(shù),
更多細(xì)節(jié)注解,見github項目提交記錄。
總結(jié)閱讀完這個函數(shù)的源碼,學(xué)習(xí)到的是浮動數(shù)與字符串的互相轉(zhuǎn)換的實現(xiàn)細(xì)節(jié),字符串與浮點數(shù)之間的關(guān)系較復(fù)雜,之后還要繼續(xù)學(xué)習(xí)。
原創(chuàng)文章,文筆有限,才疏學(xué)淺,文中若有不正之處,萬望告知。
更多精彩內(nèi)容,請關(guān)注個人公眾號。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/25882.html
摘要:而這個類型的最終之所以輸出為,是因為進(jìn)行科學(xué)計數(shù)法之后的精度丟失了,重新轉(zhuǎn)成時就恢復(fù)不了原來的值。此類問題解決方案對于存儲,超過最大表示范圍的純整數(shù),在中可以使用保存,在查詢出來的時候會將其使用類型保存的。 遇到的問題 最近遇到一個PHP大整數(shù)的問題,問題代碼是這樣的 $shopId = 17978812896666957068; var_dump($shopId); 上面的代碼輸出...
摘要:浮點數(shù)類型包括單精度浮點數(shù)和雙精度浮點數(shù)。小結(jié)通過浮點數(shù)精度的問題,了解到浮點數(shù)的小數(shù)用二進(jìn)制的表示。以后,在使用浮點數(shù)運算的時候,一定要慎之又慎,細(xì)節(jié)決定成敗。 概述 記錄下,工作中遇到的坑 ... 關(guān)于 PHP 浮點數(shù)運算,特別是金融行業(yè)、電子商務(wù)訂單管理、數(shù)據(jù)報表等相關(guān)業(yè)務(wù),利用浮點數(shù)進(jìn)行加減乘除時,稍不留神運算結(jié)果就會出現(xiàn)偏差,輕則損失幾十萬,重則會有信譽損失,甚至吃上官司,我...
摘要:有的時候,你可能有這種需求,需要將一個數(shù)字分為等份,多余的自動分配給其中一個數(shù)字。 有的時候,你可能有這種需求,需要將一個數(shù)字分為N等份,多余的自動分配給其中一個數(shù)字。 實現(xiàn)方法有如下兩種,當(dāng)然還有其他的,比如截取substr等,有興趣的可以自己嘗試: 第一種方法,采用bc函數(shù),即PHP的數(shù)學(xué)擴展庫bcmath,具體可以點擊如下鏈接查看更多了解 BC數(shù)學(xué)函數(shù) http://ph...
閱讀 979·2021-11-17 09:33
閱讀 435·2019-08-30 11:16
閱讀 2498·2019-08-29 16:05
閱讀 3374·2019-08-29 15:28
閱讀 1422·2019-08-29 11:29
閱讀 1975·2019-08-26 13:51
閱讀 3415·2019-08-26 11:55
閱讀 1238·2019-08-26 11:31