AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / unix / 问题 / 456794
Accepted
Yimin Rong
Yimin Rong
Asked: 2018-07-18 08:26:33 +0800 CST2018-07-18 08:26:33 +0800 CST 2018-07-18 08:26:33 +0800 CST

直流丢失精度

  • 772

我想dc用十六进制点处理一些基数为 16 的数字,但我遇到了精度问题。例如,下面我乘以F423F.FD,100都是十六进制。预期的答案是F423FFD,而是给出F423FFA.E1,接近但即使在四舍五入之后也不够准确。

$ dc
16 d i o F423F.FD 100 * p
F423FFA.E1

我读到那dc是一个无限精度的计算器,无论如何这都不是一个大数字。有什么我做错了吗?

感谢您的回答。考虑到 的问题dc,我硬着头皮写了自己的解析器来解析其他基数的实数。如果有人对代码感兴趣,我可以在这里发布。

dc
  • 3 3 个回答
  • 1424 Views

3 个回答

  • Voted
  1. Stephen Kitt
    2018-07-18T08:51:24+08:002018-07-18T08:51:24+08:00

    用十进制表示(dc用来转换),对应999999.98(四舍五入)×256,即255999994.88,即十六进制的F423FFA.E1。

    因此差异来自dc的舍入行为:它不是计算 256 × (999999 + 253 ÷ 256),而是将 253 ÷ 256 向下舍入并乘以结果。

    dc是一个任意精度计算器,这意味着它可以计算到你想要的任何精度,但你必须告诉它那是什么。默认情况下,它的精度为 0,这意味着除法只产生整数值,而乘法使用输入中的位数。要设置精度,请使用k(并记住精度始终以十进制数字表示,无论输入或输出基数如何):

    10 k
    16 d i o
    F423FFD 100 / p
    F423F.FD0000000
    100 * p
    F423FFD.000000000
    

    (8 位的精度就足够了,因为这是您需要用十进制表示 1 ÷ 256。)

    • 8
  2. Best Answer
    meuh
    2018-07-18T09:03:30+08:002018-07-18T09:03:30+08:00

    请注意,仅打印原始数字表明它是四舍五入的:

    $ dc <<<'16 d i o F423F.FD p'
    F423F.FA
    

    您可以通过添加大量尾随零来解决它以提高精度:

    $ dc <<<'16 d i o F423F.FD000000 100 * p'
    F423FFD.0000000
    
    • 6
  3. user232326
    2018-07-20T11:18:52+08:002018-07-20T11:18:52+08:00

    问题

    问题在于 dc(和 bc)理解数字常量的方式。
    例如,值(十六进制)0.3(除以 1)被转换为接近于0.2

    $ dc <<<"20k 16 d i o 0.3 1 / p"
    .199999999999999999999999999
    

    事实上,普通常量0.3也发生了变化:

    $ dc <<<"20 k 16 d i o     0.3     p"
    .1
    

    这似乎是一种奇怪的方式,但事实并非如此(稍后会详细介绍)。
    添加更多零会使答案接近正确值:

    $ dc <<<"20 k 16 d i o     0.30     p"
    .2E
    
    $ dc <<<"20 k 16 d i o     0.300     p"
    .2FD
    
    $ dc <<<"20 k 16 d i o     0.3000     p"
    .3000
    

    最后一个值是准确的,并且无论添加多少零都将保持准确。

    $ dc <<<"20 k 16 d i o     0.30000000     p"
    .3000000
    

    问题也存在于 bc 中:

    $ bc <<< "scale=20; obase=16; ibase=16;    0.3 / 1"
    .19999999999999999
    
    $ bc <<< "scale=20; obase=16; ibase=16;    0.30 / 1"
    .2E147AE147AE147AE
    
    $ bc <<< "scale=20; obase=16; ibase=16;    0.300 / 1"
    .2FDF3B645A1CAC083
    
    $ bc <<< "scale=20; obase=16; ibase=16;    0.3000 / 1"
    .30000000000000000
    

    一位数字?

    浮点数非常不直观的事实是所需的位数(在点之后)等于二进制位数(也在点之后)。二进制数 0.101 正好等于十进制的 0.625。二进制数 0.0001110001 (完全)等于0.1103515625(十进制数字)

    $ bc <<<'scale=30;obase=10;ibase=2; 0.101/1; 0.0001110001/1'; echo ".1234567890"
    .625000000000000000000000000000
    .110351562500000000000000000000
    .1234567890
    

    此外,对于像 2^(-10) 这样的浮点数,它在二进制中只有一个(设置)位:

    $ bc <<<"scale=20; a=2^-10; obase=2;a; obase=10; a"
    .0000000001000000000000000000000000000000000000000000000000000000000
    .00097656250000000000
    

    二进制位数.0000000001(10) 与十进制位数.0009765625(10) 相同。在其他基数中可能不是这种情况,但基数 10 是 dc 和 bc 中数字的内部表示,因此是我们真正需要关心的唯一基数。

    数学证明在这个答案的末尾。

    公元前规模

    点后面的位数可以用内置函数scale()形式 bc 计算:

    $ bc <<<'obase=16;ibase=16; a=0.FD; scale(a); a; a*100'
    2
    .FA
    FA.E1
    

    如图所示,2 位数字不足以表示常数0.FD。

    而且,仅计算点后使用的字符数是报告(和使用)数字比例的一种非常不正确的方法。数字的小数位数(任何基数)应计算所需的位数。

    十六进制浮点数中的二进制数字。

    众所周知,每个十六进制数字使用 4 位。因此,小数点后的每个十六进制数字需要 4 个二进制数字,由于上面的(奇数?)事实也需要 4 个十进制数字。

    因此,像这样的0.FD数字需要 8 个十进制数字才能正确表示:

    $ bc <<<'obase=10;ibase=16;a=0.FD000000; scale(a);a;a*100'
    8
    .98828125
    253.00000000
    

    添加零

    数学很简单(对于十六进制数字):

    • 计算点后的十六进制数字 ( h) 的数量。
    • 乘以h4。
    • 添加 h×4 - h = h × (4-1) = h × 3 = 3×h零。

    在 shell 代码中(对于 sh):

    a=F423F.FD
    h=${a##*.}
    h=${#h}
    a=$a$(printf '%0*d' $((3*h)) 0)
    echo "$a"
    
    echo "obase=16;ibase=16;$a*100" | bc
    
    echo "20 k 16 d i o $a 100 * p" | dc
    

    哪个将打印(在 dc 和 bc 中都正确):

    $  sh ./script
    F423F.FD000000
    F423FFD.0000000
    F423FFD.0000000
    

    在内部,bc(或 dc)可以使所需的位数与上面计算的数字(3*h)相匹配,以将十六进制浮点数转换为内部十进制表示。或其他基数的其他函数(假设在这种其他基数中,位数相对于基数 10(bc 和 dc 的内部)是有限的)。像 2 i (2,4,8,16,...) 和 5,10。

    正则表达式

    posix 规范指出(对于 bc,dc 基于):

    无论输入和输出基数如何,内部计算都应以十进制进行,直至指定的小数位数。

    但是“……指定的小数位数。” 可以理解为“……表示数字常量所需的十进制位数”(如上所述)而不影响“十进制内部计算”

    因为:

    bc <<<'scale=50;obase=16;ibase=16; a=0.FD; a+1'
    1.FA
    

    bc 并没有真正使用上面设置的 50(“指定的小数位数”)。

    只有在除以它时才会被转换(仍然不正确,因为它使用 2 的比例来读取常数0.FD,然后再将其扩展到 50 位):

    $ bc <<<'scale=50;obase=16;ibase=16; a=0.FD/1; a'
    .FAE147AE147AE147AE147AE147AE147AE147AE147A
    

    但是,这是准确的:

    $ bc <<<'scale=50;obase=16;ibase=16; a=0.FD000000/1; a'
    .FD0000000000000000000000000000000000000000
    

    同样,读取数字字符串(常量)应该使用正确的位数。


    数学证明

    分两步:

    二进制分数可以写成 a/2 n

    二元分数是二的负幂的有限和。

    例如:

    = 0.00110101101 = 
    = 0. 0     0      1     1      0      1     0      1      1     0       1
    

    = 0 + 0×2 -1 + 0×2 -2 + 1×2 -3 + 1×2 -4 + 0×2 -5 + 1×2 -6 + 0×2 -7 + 1×2 -8 + 1×2 -9 + 0×2 -10 + 1×2 -11

    = 2 -3 + 2 -4 + 2 -6 + 2 -8 + 2 -9 + 2 -11 =(去掉零)

    在 n 位的二进制小数中,最后一位的值为 2 -n或 1/2 n。在此示例中: 2 -11或 1/2 11。

    = 1/2 3 + 1/2 4 + 1/2 6 + 1/2 8 + 1/2 9 + 1/2 11 = (逆)

    一般来说,分母可以变成 2 n,分子的正指数为 2。然后可以将所有项组合成单个值 a/2 n。对于这个例子:

    = 2 8 /2 11 + 2 7 /2 11 + 2 5 /2 11 + 2 3 /2 11 + 2 2 /2 11 + 1/2 11 =(用 2 11表示)

    = (2 8 + 2 7 + 2 5 + 2 3 + 2 2 + 1 ) / 2 11 = (提取公因数)

    = (256 + 128 + 32 + 8 + 4 + 1) / 2 11 = (转换为值)

    = 429 / 2 11

    每个二进制分数都可以表示为 b/10 n

    a/2 n乘以 5 n /5 n,得到 (a×5 n )/(2 n ×5 n ) = (a×5 n )/10 n = b/10 n,其中 b = a×5 n . 它有n个数字。

    例如,我们有:

    (429·5 11 )/10 11 = 20947265625 / 10 11 = 0.20947265625

    已经证明,每个二进制分数都是具有相同位数的十进制分数。

    • 1

相关问题

  • 为什么 dc print 会拆分长数字

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    如何将 GPG 私钥和公钥导出到文件

    • 4 个回答
  • Marko Smith

    ssh 无法协商:“找不到匹配的密码”,正在拒绝 cbc

    • 4 个回答
  • Marko Smith

    我们如何运行存储在变量中的命令?

    • 5 个回答
  • Marko Smith

    如何配置 systemd-resolved 和 systemd-networkd 以使用本地 DNS 服务器来解析本地域和远程 DNS 服务器来解析远程域?

    • 3 个回答
  • Marko Smith

    如何卸载内核模块“nvidia-drm”?

    • 13 个回答
  • Marko Smith

    dist-upgrade 后 Kali Linux 中的 apt-get update 错误 [重复]

    • 2 个回答
  • Marko Smith

    如何从 systemctl 服务日志中查看最新的 x 行

    • 5 个回答
  • Marko Smith

    Nano - 跳转到文件末尾

    • 8 个回答
  • Marko Smith

    grub 错误:你需要先加载内核

    • 4 个回答
  • Marko Smith

    如何下载软件包而不是使用 apt-get 命令安装它?

    • 7 个回答
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Wong Jia Hau ssh-add 返回:“连接代理时出错:没有这样的文件或目录” 2018-08-24 23:28:13 +0800 CST
  • Martin Hope
    Evan Carroll systemctl 状态显示:“状态:降级” 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim 我们如何运行存储在变量中的命令? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S 为什么 /dev/null 是一个文件?为什么它的功能不作为一个简单的程序来实现? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 如何从 systemctl 服务日志中查看最新的 x 行 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - 跳转到文件末尾 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla 为什么真假这么大? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis 在一个巨大的(70GB)、一行、文本文件中替换字符串 2017-12-30 06:58:33 +0800 CST
  • Martin Hope
    Bagas Sanjaya 为什么 Linux 使用 LF 作为换行符? 2017-12-20 05:48:21 +0800 CST

热门标签

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve