[問題] json大數字變成了科學符號

看板C_and_CPP (C/C++)作者 (竹科學友哥)時間1年前 (2022/08/02 17:31), 1年前編輯推噓3(3023)
留言26則, 5人參與, 1年前最新討論串1/1
開發平台(Platform): (Ex: Win10, Linux, ...) Win10 編譯器(Ex: GCC, clang, VC++...)+目標環境(跟開發平台不同的話需列出) NetBeans + Linux 額外使用到的函數庫(Library Used): (Ex: OpenGL, ...) cJson (https://github.com/DaveGamble/cJSON) 問題(Question): 如果json的資料是 { "str": "4761739001010135", } 讀取後,將字串轉成數字,使用cJSON_AddNumberToObject後 如預期的得到 { "num": 4761739001010135 } 但是遇到尾數是零時,相同的程式碼就會變成科學符號 例如,如果json的資料是 { "str": "4761739001010010" } 結果得到 (非預期) { "num": 4.76173900101001e+15 } 相關程式碼如下: cJSON *pjNode = NULL; pjNode = cJSON_GetObjectItem(pInJson, "str"); if (pjNode != NULL) { cJSON_AddNumberToObject(pOutJson, "num", atoll(pjNode->valuestring)); } 感覺是大數字才會發生問題,因此想請教一下 1. 為何相同的程式碼,當尾數是0時,會變成科學符號呢? 2. 如何避免/修改成一般的數字,而不是科學符號呢? 麻煩了 感謝 餵入的資料(Input): 預期的正確結果(Expected Output): 錯誤結果(Wrong Output): 程式碼(Code):(請善用置底文網頁, 記得排版,禁止使用圖檔) 補充說明(Supplement): -- ※ 發信站: 批踢踢實業坊(ptt.cc), 來自: 60.250.189.43 (臺灣) ※ 文章網址: https://www.ptt.cc/bbs/C_and_CPP/M.1659432682.A.B48.html ※ 編輯: renmax (60.250.189.43 臺灣), 08/02/2022 17:32:03

08/02 18:06, 1年前 , 1F

08/03 16:30, 1年前 , 2F
C大 我是使用cJSON_AddNumberToObject後 去print json內容
08/03 16:30, 2F

08/03 16:31, 1年前 , 3F
發現 原本print出來的應該是長整數 但卻顯示科學符號
08/03 16:31, 3F

08/03 16:32, 1年前 , 4F
而科學符號 並不是我想要的結果 我希望print出來後 是整數
08/03 16:32, 4F

08/03 16:33, 1年前 , 5F
型態 您給的參考網址是取值 而不是塞值 謝謝幫忙>"<
08/03 16:33, 5F

08/03 16:57, 1年前 , 6F
cJSON.c 找 Try 15 decimal 看那邊. %1.15g 和 %1.17g 的差別
08/03 16:57, 6F

08/03 17:12, 1年前 , 7F
這實作似乎不符合 RFC 8259 第六節 Numbers 的最後一段.
08/03 17:12, 7F

08/03 17:18, 1年前 , 8F
可能只參照到 RFC 4627.
08/03 17:18, 8F

08/04 13:44, 1年前 , 9F
人家就說cJSON不支援long long,跟取值還是塞值沒關係。
08/04 13:44, 9F

08/04 14:15, 1年前 , 10F
不好意思 如果不支援long long的話 請問一下
08/04 14:15, 10F

08/04 14:16, 1年前 , 11F
1. 上方的例子中 為何只有尾數是零 才會變成科學符號呢?
08/04 14:16, 11F

08/04 14:19, 1年前 , 12F
2. 有什麼方法 可以做到預期的結果呢?(整數長度MAX = 20)
08/04 14:19, 12F

08/04 14:20, 1年前 , 13F
感謝y大和C大幫忙
08/04 14:20, 13F

08/04 15:44, 1年前 , 14F
1. 上面yvb已經回答了 2. 沒有,請看完上面的issue回答
08/04 15:44, 14F

08/05 04:43, 1年前 , 15F
51 bits換算是15.3位,長度20已經超過了,cJSON不能用。
08/05 04:43, 15F

08/05 04:47, 1年前 , 16F
只能找其他的或是fork cJSON自己修
08/05 04:47, 16F

08/05 20:30, 1年前 , 17F
長度 20 也不能用 double 或 int64_t 來存
08/05 20:30, 17F

08/05 20:31, 1年前 , 18F
uint64_t 只能擦邊 (18446744073709551615 正好 20 位)
08/05 20:31, 18F

08/05 20:31, 1年前 , 19F
所以任意 20 位的數字可能要建議當字串來用
08/05 20:31, 19F

08/08 15:26, 1年前 , 20F
感謝各位大大的解釋 我仍然不太懂y大說的Try 15 decimal這
08/08 15:26, 20F

08/08 15:27, 1年前 , 21F
邊的意思 為何特定使用15和17 decimal來做判斷呢?
08/08 15:27, 21F

08/09 03:16, 1年前 , 22F
double 的精確度是 53 bit, 數學轉換對應十進位約 15.95 位
08/09 03:16, 22F

08/09 03:16, 1年前 , 23F
所以一般可能會認為印出 1+15 位數就能足夠表示一個 double
08/09 03:16, 23F

08/09 03:17, 1年前 , 24F
但實際上如果要達成所謂的 round-trip (轉成字串再轉回來時
08/09 03:17, 24F

08/09 03:17, 1年前 , 25F
要得到完全一樣的 double 數值) 會需要印出 1+17 位
08/09 03:17, 25F

08/09 03:17, 1年前 , 26F
才能使某些邊界狀況能夠得到分辨
08/09 03:17, 26F
文章代碼(AID): #1YwExgj8 (C_and_CPP)
文章代碼(AID): #1YwExgj8 (C_and_CPP)