你真的搞懂了负数取模吗?

作者: 志哥谈笑间 | 来源:发表于2018-07-10 09:26 被阅读6次

如果现在给你出下面这道面试题,你能答出来吗?

-7 % 3 = ?

那这个呢?

7 % (-3) = ?

正整数的取模大家应该玩的很溜了,(什么,正数的都不会,那自己去谷歌吧,百度也行。) 对于负数呢,上周有人在一个群里问这个问题,我以为我是会的,后来发现我的答案是错的,索性就去研究了一番。

带着问题我打开了维基百科 https://en.wikipedia.org/wiki/Modulo_operation ,看到了这段话。

在几乎所有的计算系统中,取模运算都满足下面这个公式:

a = nq + r  |r| < |a|
假设 q 是 a、b 相除得到的商(quotient),r 是相应的余数(remainder)

然而,取模操作又依赖于编程语言和底层硬件。

我们现在用这个工具地址 https://c.runoob.com/ 打开我学习的语言,比如 Java/C,会发现

-7 % 3 = -1
7 % (-3) = 1 

既然编程语言决定了结果,那换一种语言,比如 Python,发现结果竟然真的不一样,你说神奇不神奇。

-7 % 3 = 2
7 % (-3) = -2

这是为什么呢?

我们知道,当余数不为 0 的时候,取整除(符号是「/」,叫法不一样,可以纠正我一下),可能出现多种方式,下面是一些你应该知道的例子。

向上取整。向 +∞ 方向取最接近精确值的整数,也就是取比实际结果稍大的最小整数,也叫 Ceiling 取整。这种取整方式下,

17 / 10 = 2,5 / 2 = 3, -9 / 4 = -2

向下取整。向 -∞ 方向取最接近精确值的整数,也就是取比实际结果稍小的最大整数,也叫 Floor 取整。这种取整方式下,

17 / 10 = 1,5 / 2 = 2, -9 / 4 = -3

向零取整。向 0 方向取最接近精确值的整数,换言之就是舍去小数部分,因此又称截断取整(Truncate)。这种取整方式下,

17 / 10 = 1,5 / 2 = 2, -9 / 4 = -2

我们再来看刚才的公式

a = nq + r 
我们知道商 q = a/n,从而得出
r = a - (a/n) * n
而 (a/n) 这个结果取决于上面几种方式用哪个

下面我们就重点看这个 (a/n)

常用的计算机语言用的除法方式是下面这两种,truncate 除法floor 除法

truncate 除法 即是上面的向零取整,也叫趋零截尾,而 floor 除法 即是上面的向下取整,也叫趋负无穷截尾。

而 Java/C 等语言用的是 truncate 除法,Python 用的是 floor 除法。

这下就明白了吧,现在我们再来看开始的面试题。

在 Java/C 中,

-7 % 3 = -1
7 % (-3) = 1 
// 下面是推倒过程
-7 % 3 = -7 - trunc(-7/3) * 3 = -7 - (-2) * 3 = -7 + 6 = -1
7 % (-3) = 7 - trunc(7 / (-3)) * (-3) = 7 - (-2) * (-3) = 7 - 6 = 1

在 Python 中,

-7 % 3 = 2
7 % (-3) = -2
## 下面是推倒过程
-7 % 3 = -7 - floor(-7/3) * 3 = -7 - (-3) * 3 = -7 + 9 = 2
7 % (-3) =  7 - floor(7 / (-3)) * (-3) = 7 - (-3) * (-3) = 7 - 9 = -2

如果是其他语言,先去搞清楚是用的那种方式。

本文参考:
Modulo operation:
https://en.wikipedia.org/wiki/Modulo_operation
负数取模怎么算:
https://www.jianshu.com/p/452c1a5acd31

如果觉得内容还不错,可以关注一下我哦
期待与你相遇,一同成长前行!

相关文章

  • 你真的搞懂了负数取模吗?

    如果现在给你出下面这道面试题,你能答出来吗? 那这个呢? 正整数的取模大家应该玩的很溜了,(什么,正数的都不会,那...

  • 负数取模怎么算

    对于整数的取模运算,想必大家已经比较熟悉了,譬如说 7 对 3 取模,结果是多少,我们可以按照小学的公式:被除数÷...

  • 取模

    取模和求余主要区别在负数。 c语言中求余,-1%255=-1;python中取模,-1%255=254.

  • 16-整数次方

    底可能为0, 指数可能为负数,负数取余还是负数,例如 -3 % 2 = -1

  • swift - 取模(浮点取模)

    浮点型取模 let rem = 2.5 % 1.1 print(rem) // 0.3In Swift 3+ ...

  • C语言学习笔记

    负数求模运算符号决定C99规定"趋零截断".如果第一个运算数为负数,那么求模结果为负数;如果第一个运算数是正数,那...

  • 知识点不会融合怎么办?

    说到知识点,人人都会说:我学了呀,会了呀,一问到:懂了吗?懂了。真的懂了吗?真的懂了呀。确定吗?确定啊。那你把这个...

  • 如何阅读?20170213

    你真的会阅读吗?你读过的书真的读懂了吗?你理解了作者想要表达的概念、故事、逻辑了?你真的会选书吗?什么样的书才适合...

  • 如何让更多的人升级自己的操作系统,实现财富自由

    很多人都在说,我懂了那么多道理,为什么依然过不好这一生?其实你是真的懂了那些道理吗?即便是懂了那些道理,你有运用这...

  • C语言学习 - 求余(取模)运算余数符号

    在C语言中,求余运算,又称取模运算,其余数符号取决于被除数,即被除数为正数,则余数为正数;被除数为负数,则余数为负...

网友评论

    本文标题:你真的搞懂了负数取模吗?

    本文链接:https://www.haomeiwen.com/subject/gjtfpftx.html