程序员视角看手机计算器上为什么10%+10% != 0.2 ?

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。

本文链接:https://www.cnblogs.com/lihuidashen/p/11543960.html

 

https://mp.weixin.qq.com/s/MnZFWk_yYN67jSq7fruhHg

争议话题

    近日,一则热搜#手机计算器全线阵亡#的话题在网上火了起来。不少网友惊奇地发现,在自己的智能手机上打开计算器,计算10%+10%,得出的结果竟然是0.11!

 

    网友反映,华为、苹果、OPPO、VIVO、小米、一加等多个品牌的手机计算器都出现了这样的“BUG”;也有人发现,魅族、锤子科技、努比亚手机的计算器结果是0.2,贴上了很多逻辑思维,文化差异的标签,下面用两个手机对比一下差异。

 

 

 

为什么会出现这样的“BUG”?

    不少科技博主指出,出现这样的“BUG”,原因主要是计算器对输入符号的理解不同。一些手机厂商的高管也出来进行了解释,比如魅族科技的副总裁华海良在微博的发言:

 

 

 

    还有荣耀业务部总裁赵明,也在微博发表了自己的观点,很有意思。


 

 

    没想到在18年的时候,MIUI就回复了这个问题,现在才火起来,估计当时都是懵懵的,不明觉厉。

 

 

 

 

 

官方解释

    我们在进行四则运算时,都是先乘除后加减,如果有括号就先算括号里的。而百分号“%”则代表“除以100”,和乘除是同样的优先级。因此,我们计算10%+10%得出的结果为0.2。即10%+10%=0.1+0.1=0.2。

 

    在传统的百分号运算中,计算器对输入符号的理解和我们是不一样的。传统的百分号运算都是在第一个数的基础上增加10%作为第二个数。也就是X+n%就是在X的基础上上浮n%,也就是X+X*n%。在10%+10%的运算中,计算过程即为10%+10%=10%+10%*10%=10%*(1+10%)=0.11。

 

从程序员角度来看代码

    作为一个程序员,自然要从代码的角度来看了。Windows Calculator应用程序是一个用 C ++编写的现代Windows应用程序,预装了Windows。该应用程序提供标准,科学和程序员计算器功能,以及各种度量单位和货币之间的一组转换器,其中只有标准模式有“%”。

 

 

 

 

    为此我在github上找到了微软开源的计算器项目,我找到关于“%”计算的部分,摘出了其中相关的代码:

case IDC_PERCENT:
{
    // If the operator is multiply/divide, we evaluate this as "X [op] (Y%)"
    // Otherwise, we evaluate it as "X [op] (X * Y%)"
    if (m_nOpCode == IDC_MUL || m_nOpCode == IDC_DIV)
    {
        result = rat / 100;
    }
    else
    {
        result = rat * (m_lastVal / 100);
    }
    break;
}

 

 

    注释中也已经解释了,当操作符是乘法或者除法的时候,与“%”相关的直接除以100再和另外的数操作(即我们通常认识的算法),否则就按照上一次结果的百分比来计算。

    所以,如果你计算10%+10%,它是下面的过程:

 

结果 —— 操作

0    初始值0    输入10%,计算0 + 10% * 00    输入+10%,计算0 + 10 *0

 

最终会得到0

    只不过很多手机计算器中直接把第一个10%当成了0.1,这也就是我们看到一些手机计算器最终会得到0.11结果的原因

    但是如果你计算1000 * 10%,它按照原始的方式计算,即计算得到100。所以这是有意为之,而并非什么bug!程序员表示不背这个锅。

 

    另外我们都知道,“%”常用于取模运算,它是一个二元运算符,这也正是科学模式和程序员模式没有“%”的原因,因为“Mod”取代了,例如:7%2 = 1

 

 所以当你在Linux的命令行输入bc,然后输入10+10%,你会看到下面的结果:

$ bc
10+10%
(standard_in) 3: syntax error
7%2
1

  没错,它会提示你语法错误,而不是帮你计算10的10%,因为这里的“%”并非计算百分数,而是用来取模的。所以在windows自带的程序员计算器和科学计算器中,有“Mod”,而没有“%”。

 

注意:Linux下,bc命令是一种支持任意精度的交互执行的计算器语言,可以很方便的进行浮点运算,当然整数运算也不再话下。

 

另类解决办法

    如果在输入时,将每个10%乘1,就能得到正确答案,至于为什么会这样,就交给大家去思考吧。

 

总结

    从程序员视角看,这不是Bug,“%”在不同的场景之下有不一样的作用,人们也在不断更新进步,将这些生活场景应用在机器软件上,智能的路还有很远,但就在未来。

 

讨论一下吧

  • 你的手机在不同模式下是什么样的呢?

  • 这种计算方式有必要吗?

  • 身为一名程序员,这个锅你愿意背吗?

 

推荐阅读

(点击标题可跳转阅读)

Qt 学习笔记-强势入门

Qt 学习笔记-Qt中添加背景图片的方法

Qt 学习笔记-处理鼠标响应事件

C++小游戏:扑克牌21点

const 指针与指向const的指针

hello world 程序是如何被编译出来的?

C语言main函数有哪些写法呢

【编程之美】用C语言实现状态机(实用)

【超详细C语言】带你吃透贪吃蛇游戏之精髓

 

关注公众号【技术让梦想更伟大】,获取更多Linux/C/C++/Python/FPGA等原创技术文章。后台免费获取经典电子书籍和视频资源,实时更新,原创不易,请多支持,谢谢!

 

posted @ 2019-09-18 17:33  技术让梦想更伟大  阅读(1658)  评论(0编辑  收藏  举报