【C语言的小角落】--- 深度理解取余/取模运算

 Welcome to 9ilk's Code World

       

(๑•́ ₃ •̀๑) 个人主页:       9ilk

(๑•́ ₃ •̀๑) 文章专栏:     C语言的小角落 


本篇博客我们来深度理解取余/取模,以及它们在不同语言中出现不同现象的原因。


🏠 关于取整

🎵 向0取整

#include <stdio.h>
#include <windows.h>
int main()
{
  //本质是向0取整
  int i = -2.9;
  int j = 2.9;
  printf("%d\n", i); //结果是:-2
  printf("%d\n", j); //结果是:2
  int a = 5;
  int b = -5;
  printf("%d %d",a/2,b/2);
  system("pause");
  return 0;
}

测试结果:

我们发现测试结果中浮点数取整都是往0方向取整的:

其实在C库中有个trunc取整函数,也是向0取整:

🎵 向-∞取整

#include <stdio.h>
#include <math.h> //因为使用了floor函数,需要添加该头文件
#include <windows.h>
int main()
{
 //本质是向-∞取整,注意输出格式要不然看不到结果
  printf("%.1f\n", floor(-2.9)); //-3
  printf("%.1f\n", floor(-2.1)); //-3
  printf("%.1f\n", floor(2.9)); //2
  printf("%.1f\n", floor(2.1)); //2
  system("pause");
  return 0;
}

测试结果:

我们发现当调用floor函数取整时是结果都变小,往-∞方向取整

🎵 向+∞取整

#include <stdio.h>
#include <math.h>
#include <windows.h>
int main()
{
  //本质是向+∞取整,注意输出格式要不然看不到结果
  printf("%.1f\n", ceil(-2.9)); //-2
  printf("%.1f\n", ceil(-2.1)); //-2
  printf("%.1f\n", ceil(2.9)); //3
  printf("%.1f\n", ceil(2.1)); //3
  system("pause");
  return 0;
}

测试结果:

我们发现调用ceil函数时取整是结果都变大,往+∞方向取整

🎵 四舍五入取整

#include <stdio.h>
#include <math.h>
#include <windows.h>
int main()
{
 //本质是四舍五入
 printf("%.1f\n", round(2.1));
 printf("%.1f\n", round(2.9));
 printf("%.1f\n", round(-2.1));
 printf("%.1f\n", round(-2.9));
 system("pause");
 return 0;
}

测试结果:

round()函数采用的取整方式就是我们四舍五入取整,逢五进一。

🎵 多种取整方式汇总

#
include <stdio.h>
#include <math.h>
#include <windows.h>
int main()
{
  const char * format = "%.1f \t%.1f \t%.1f \t%.1f \t%.1f\n";
  printf("value\tround\tfloor\tceil\ttrunc\n");
  printf("-----\t-----\t-----\t----\t-----\n");
  printf(format, 2.3, round(2.3), floor(2.3), ceil(2.3), trunc(2.3));
  printf(format, 3.8, round(3.8), floor(3.8), ceil(3.8), trunc(3.8));
  printf(format, 5.5, round(5.5), floor(5.5), ceil(5.5), trunc(5.5));
  printf(format, -2.3, round(-2.3), floor(-2.3), ceil(-2.3), trunc(-2.3));
  printf(format, -3.8, round(-3.8), floor(-3.8), ceil(-3.8), trunc(-3.8));
  printf(format, -5.5, round(-5.5), floor(-5.5), ceil(-5.5), trunc(-5.5));
  system("pause");
  return 0;
}

测试结果:

总结一下:

  • 浮点数(整数/整数),是有很多的取整方式的。常见的是向上取整,向下取整,向0取整,四舍五入取整,其中C语言默认是向0取整
  • 从汇总例子看,相同的浮点数采用不同的取整方案也可能得到相同的整数
  • 取整方案的使用取决于你的具体场景,比如每台服务器能处理 5 个任务,18 个任务需要 ⌈18/5⌉=4 台服务器,用向上取整预留足够的额外资源。

🏠 关于取模

🎵 取模初步概念

如果a和d是两个自然数,d非零,可以证明存在两个唯一的整数 q 和 r,满足 a = q*d + r 0 ≤ r < d。其中,q被称为商,r 被称为余数。

#include <stdio.h>
#include <windows.h>
int main()
{
 int a = 10;
 int d = 3;
 printf("%d\n", a%d); //结果是1
 system("pause");
 return 0;
}

测试结果:

测试结果是符合我们给的定义的,因为a=10,d=3,q=3,r=1(其中 0<= r < d),因此a = q*d + r -> 10 = 3*3 +1。

如果是以下测试代码呢?

🎵 取模修订定义

  • C语言 vs2019
#include<stdio.h>
#include<windows.h>
#include<math.h>

int main()
{
	int a = -10;
	int d = 3;
	//printf("%d\n", a/d); //C语言中是-3,很好理解
	printf("%d\n", a % d);
	return 0;
}

测试结果:

  • Python 3.10.12 

很显然,上面关于取模的定义,并不能满足语言上的取模运算 : 在C语言测试环境下,10%3得到的余数不满足r>0的要求。因此大家对取模有了一个修订版定义:

如果a和d是两个自然数,d非零,可以证明存在两个唯一的整数 q 和 r,满足 a = q*d + r , q 为整数,且0 ≤ |r|< |d|。其中,q 被称为商,r 被称为余数。

对于C:-10 = (-3) * 3 + (-1) ,其中0 < |-1| <  3

对于Python:-10 = (?) * 3 + 2,可以推导出其q应该为-4才能满足定义。

  • 在不同语言中,同一个计算表达式,负数“取模”结果是不同的,我们可以称之为正余数负余数

Q:为什么会出现正余数和负余数?

答:具体余数r的大小,本质是取决于商q的。而商,取决于除法计算时的取整规则!因此本质是处理商时的取整方式不同,导致了得到的具体余数不同!

🏠 取模 vs 取余

取模和取余两者并不能严格等价(但是大部分情况下是能等价的),取余或者取模,都应该要算出商,然后才能得出余数。在计算机科学中,我们规定的取余和取模本质上它们的取整方式不同

  • 取余:尽可能让商,进行向0取整。
  • 取模:尽可能让商,进行向-∞取整。

因此我们可以推导出:

1. 对任何一个大于0的数(正数),对其进行0向取整和-∞取整,取整方向一致(方向都指向横向数轴左边),此时取模等于取余

2. 对任何一个小于0的数(负数),对其进行0向取整和-∞取整,取整方向相反(0向取整指向右,负无穷取整指向左),此时取模不等于取余

🎵 同符号运算

  • C语言
#include <stdio.h>
#include <windows.h>
int main()
{
    printf("被除数和除数都是正数:\n");
	printf("%d\n", 10 / 3);
	printf("%d\n", 10 % 3);
	printf("被除数和除数都是负数:\n");
	printf("%d\n", -10 / -3);
	printf("%d\n", -10 % -3);
	system("pause");
    return 0;
}

测试结果:

  • Python 3.10.12
print(10//3)
print(10%3)
print(-10//-3)
print(-10%-3)
注意:python中 / 默认是浮点数除法,//才是整数除法,并进行-∞取整

测试结果:

通过不同环境对比,我们发现当同符号数据相除时,它们的商一定是正数(正数vs正整数),即大于0!因此,在对其商进行取整时,取模等价于取余

  • 结论:参与相除的两个数据,如果同符号,取模等价于取余!

🎵 不同符号运算

  • C语言
#include <windows.h>
#include<stdio.h>
int main()
{
 printf("%d\n", -10 / 3); //结果:-3
 printf("%d\n\n", -10 % 3); //结果:-1 为什么? -10=(-3)*3+(-1)
 printf("%d\n", 10 / -3); //结果:-3
 printf("%d\n\n", 10 % -3); //结果:1 为什么?10=(-3)*(-3)+1
 system("pause");
 return 0;
}

测试结果:

不同符号,余数求法可以参照之前定义,在C中余数符号和被除数相同,那在Python环境呢?

  • Python 3.10.12

我们发现在Python中余数符号和除数相同,为什么和C语言会产生不同?

理解:

1. a = q*d + r (q为商,r为余数)变换成 r = a - q*d = a + (-q*d)。

2. 我们知道对于x =  y + z这样的表达式,x的符号与|y|和|z|中较大的一致;因此r也就是余数的符号就取决于|a|和|-q*d|谁大,被除数a是固定的,那么就取决于商q的取整方式!

3. C是向0取整,因此商q本身绝对值是减小的,此时由于本来0<|r|<|d|,那此时就是|a|大,即余数符号由被除数决定。(简单理解,略有不严谨)

4. Python是向-∞取整,因此商q本身绝对值是增大的,此时大概就是|-q*d|大一些,即余数符号由除数d决定。

结论:如果参与取余的两个数据符号不同,在C语言中(或其他采用0向取整的语言),余数符号与被除数相同;而采用负无穷取整的语言,余数和除数相同。

🏠 总结

1. 浮点数(或者整数相除),是有很多的取整方式的。

2. 取模修正定义:如果a和d是两个自然数,d非零,可以证明存在两个唯一的整数 q 和 r,满足 a = q*d + r , q 为整数,且0 ≤ |r| < |d|。其中,q 被称为商,r 被称为余数。

3. 在不同语言,同一个计算表达式,“取模”结果是不同的。我们可以称之为分别叫做正余数和负余数;具体余数r的大小,本质是取决于商q的。而商,又取决于除法计算的时候的取整规则

4. 取余vs取模: 取余尽可能让商,进行向0取整取模尽可能让商,向-∞方向取整;对于正数取模等价取余,对于负数,取模与取余不等价

5. 如果参与取余的两个数据符号不同,在C语言中(或其他采用0向取整的语言),余数符号与被除数相同;而采用负无穷取整的语言,余数符号和除数相同


完。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/947497.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

mapbox进阶,添加路径规划控件

👨‍⚕️ 主页: gis分享者 👨‍⚕️ 感谢各位大佬 点赞👍 收藏⭐ 留言📝 加关注✅! 👨‍⚕️ 收录于专栏:mapbox 从入门到精通 文章目录 一、🍀前言1.1 ☘️mapboxgl.Map 地图对象1.2 ☘️MapboxDirections 控件二、🍀添加路径规划控件1. ☘️实现思路2. ☘️…

linux-25 文件管理(三)复制、移动文件,cp,mv

命令cp是copy的简写&#xff0c;而mv则是move的简写。那既然copy是用于实现复制文件的&#xff0c;那通常一般我们要指定其要复制的是谁&#xff1f;而且复制完以后保存在什么地方&#xff0c;对吧&#xff1f;那因此它的使用格式很简单&#xff0c;那就是cp srcfile dest&…

IDEA开发Java应用的初始化设置

一、插件安装 如下图所示&#xff1a; 1、Alibaba Java Coding Guidelines 2.1.1 阿里开发者规范&#xff0c;可以帮忙本地自动扫描出不符合开发者规范的代码&#xff0c;甚至是代码漏洞提示。 右击项目&#xff0c;选择《编码规约扫描》&#xff0c;可以进行本地代码规范扫…

GPU加速计算的专业云服务平台:蓝耘GPU算力平台的概述、具体应用与教学

文章目录 一、平台介绍蓝耘GPU算力平台概述平台优势与特点 二、注册与登录账号注册流程GPU服务器类型配置选择指南内存和存储容量网络带宽CPU配置 三、创建实例**实例创建步骤**镜像选择与设置 四、连接实例SSH连接方法远程桌面配置 一、平台介绍 蓝耘GPU算力平台概述 蓝耘GP…

golang:微服务架构下的日志追踪系统(二)

背景 在使用Gin框架进行服务开发时&#xff0c;我们遇到了一个日志记录的问题。由于Gin的上下文&#xff08;*gin.Context&#xff09;实现了context.Context接口&#xff0c;在调用日志记录器的Info、Warn、Error等方法时&#xff0c;直接传递Gin的上下文通常不会导致编译错误…

Vue项目整合与优化

前几篇文章&#xff0c;我们讲述了 Vue 项目构建的整体流程&#xff0c;从无到有的实现了单页和多页应用的功能配置&#xff0c;但在实现的过程中不乏一些可以整合的功能点及可行性的优化方案&#xff0c;就像大楼造完需要进行最后的项目验收改进一样&#xff0c;有待我们进一步…

网关的介绍

网关&#xff08;Gateway&#xff09;在网络技术中扮演着举足轻重的角色。为了让你更好地理解网关及其相关术语&#xff0c;我会尽量用简洁明了的语言来解释&#xff0c;同时也会穿插一些专业术语以便你深入学习。 网关的基本概念 网关&#xff0c;顾名思义&#xff0c;是网络的…

【C语言程序设计——循环程序设计】枚举法换硬币(头歌实践教学平台习题)【合集】

目录&#x1f60b; 任务描述 相关知识 一、循环控制 / 跳转语句的使用 1. 循环控制语句&#xff08;for 循环&#xff09; 2. 循环控制语句&#xff08;while 循环&#xff09; 3. 跳转语句&#xff08;break 语句&#xff09; 4. 跳转语句&#xff08;continue 语句&…

SD-WAN怎样减少异地组网的网络延迟?

在经济全球化的推动下&#xff0c;许多企业的业务已经扩展到多个国家或地区。这种情况下&#xff0c;企业需要搭建高效、稳定的网络连接&#xff0c;以确保异地的分支机构之间能够顺畅地交流。网络延迟是拉低异地组网数据传输效率的重要因素&#xff0c;直接影响到企业的运营和…

小程序学习06——uniapp组件常规引入和easycom引入语法

目录 一 组件注册 1.1 组件全局注册 1.2 组件全局引入 1.3 组件局部引入 页面引入组件方式 1.3.1 传统vue规范&#xff1a; 1.3.2 通过uni-app的easycom 二 组件的类型 2.1 基础组件列表 一 组件注册 1.1 组件全局注册 &#xff08;a&#xff09;新建compoents文件…

uniapp 微信小程序 自定义日历组件

效果图 功能&#xff1a;可以记录当天是否有某些任务或者某些记录 具体使用&#xff1a; 子组件代码 <template><view class"Accumulate"><view class"bx"><view class"bxx"><view class"plank"><…

上升沿下降沿递增

沿指令&#xff1a;P&#xff1a;上升沿 从01 导通一个扫描周期 N&#xff1a;下降沿 从10 导通一个扫描周期

大数据-268 实时数仓 - ODS层 将 Kafka 中的维度表写入 DIM

点一下关注吧&#xff01;&#xff01;&#xff01;非常感谢&#xff01;&#xff01;持续更新&#xff01;&#xff01;&#xff01; Java篇开始了&#xff01; MyBatis 更新完毕目前开始更新 Spring&#xff0c;一起深入浅出&#xff01; 目前已经更新到了&#xff1a; H…

微博_14.12.2-内置猪手 会员版

微博猪手是一款作用于微博的 XposedLsposed 模块&#xff0c;可以支持未root用户和已root用户使用。进入【我的】页面&#xff0c;点击【右上角的设置】&#xff0c;点击【微博猪手】即可进一步设置其他功能。通过微博猪手模块可以实现去除各种广告&#xff08;开屏、信息流等&…

计算机网络 (21)网络层的几个重要概念

前言 计算机网络中的网络层是OSI&#xff08;开放系统互连&#xff09;模型中的第三层&#xff0c;也是TCP/IP模型中的第二层&#xff0c;它位于数据链路层和传输层之间&#xff0c;负责数据包从源主机到目的主机的路径选择和数据转发。 一、网络层的主要功能 路由选择&#xf…

openwrt nginx UCI配置过程

openwrt 中nginx有2种配置方法&#xff0c;uci nginx uci /etc/config/nginx 如下&#xff1a; option uci_enable true‘ 如果是true就是使用UCI配置&#xff0c;如果 是false&#xff0c;就要使用/etc/nginx/nginx.conf&#xff0c;一般不要修改。 如果用UCI&#xff0c;其…

【深度学习进阶】基于CNN的猫狗图片分类项目

介绍 基于卷积神经网络&#xff08;CNN&#xff09;的猫狗图片分类项目是机器学习领域中的一种常见任务&#xff0c;它涉及图像处理和深度学习技术。以下是该项目的技术点和流程介绍&#xff1a; 技术点 卷积神经网络 (CNN): CNN 是一种专门用于处理具有类似网格结构的数据的…

uni-app 页面生命周期及组件生命周期汇总(Vue2、Vue3)

文章目录 一、前言&#x1f343;二、页面生命周期三、Vue2 页面及组件生命周期流程图四、Vue3 页面及组件生命周期流程图4.1 页面加载时序介绍4.2 页面加载常见问题4.3 onShow 和 onHide4.4 onInit4.5 onLoad4.6 onReachBottom4.7 onPageScroll4.8 onBackPress4.9 onTabItemTap…

缓存淘汰算法:次数除以时间差

记录缓存中的每一项的访问次数、最后访问时间&#xff0c;获取当前时间&#xff0c;可算出时间差&#xff0c;然后&#xff0c;用次数除以时间差&#xff0c;取最小的淘汰。 这一算法比较慢&#xff0c;需配合多级缓存。一级缓存不很大&#xff0c;使用此算法。二级缓存可以大…

uniapp 微信小程序开发使用高德地图、腾讯地图

一、高德地图 1.注册高德地图开放平台账号 &#xff08;1&#xff09;创建应用 这个key 第3步骤&#xff0c;配置到项目中locationGps.js 2.下载高德地图微信小程序插件 &#xff08;1&#xff09;下载地址 高德地图API | 微信小程序插件 &#xff08;2&#xff09;引入项目…