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

news2025/1/6 8:36:39

 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向取整的语言),余数符号与被除数相同;而采用负无穷取整的语言,余数符号和除数相同


完。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

网关的主要类型和它们的特点

网关&#xff0c;作为网络通信的关键节点&#xff0c;根据其应用场景和功能特点&#xff0c;可以分为多种类型。 1.协议网关 特点&#xff1a; • 协议转换&#xff1a;协议网关的核心功能是转换不同网络之间的通信协议。例如&#xff0c;它可以将IPv4协议的数据包转换为IPv6协…

聆听音乐 1.5.9 | 畅听全网音乐,支持无损音质下载

聆听音乐手机版是面向广大音乐爱好者的移动应用程序&#xff0c;用户可以随时随地通过手机享受丰富的音乐资源。它提供了多种魅力功能&#xff0c;让用户在手机上畅享更舒适的音乐体验&#xff0c;每位用户都能享受精彩纷呈的收听体验。此外&#xff0c;软件还支持无损音质音乐…

实现一个通用的树形结构构建工具

文章目录 1. 前言2. 树结构3. 具体实现逻辑3.1 TreeNode3.2 TreeUtils3.3 例子 4. 小结 1. 前言 树结构的生成在项目中应该都比较常见&#xff0c;比如部门结构树的生成&#xff0c;目录结构树的生成&#xff0c;但是大家有没有想过&#xff0c;如果在一个项目中有多个树结构&…

“善弈者”也需妙手,Oclean欧可林:差异化不是说说而已

作者 | 曾响铃 文 | 响铃说 俗话说&#xff0c;“牙痛不是病&#xff0c;痛起来要人命”。这话意思大家都知道&#xff0c;牙痛虽不是什么大病&#xff0c;可一旦发作却是极难忍受。 前几日&#xff0c;Oclean欧可林举办了一场AirPump A10氧气啵啵冲牙器新品品鉴会&#xff…

汇编语言与接口技术--跑马灯

一、 实验要求 在单片机开发板的LED灯D1~D8上实现跑马灯。LED与单片机引脚连线电路如下图: 单片机芯片选择AT89C51&#xff0c;晶振频率设为12MHz&#xff0c;操作参考单片机开发板使用说明。跑马灯点亮的时间间隔约为1秒。分别用定时器的模式1和模式2实现。&#xff08;用P83…

基于数据融合的智能家居环境监测系统研究与设计(论文+源码)

1总体方案设计 本次基于数据融合的智能家居环境监测系统的设计&#xff0c;其系统总体架构如图2.1所示&#xff0c;整个系统在器件上包括了主控制器STM32F103单片机&#xff0c;MQ可燃气体传感器&#xff0c;光照传感器&#xff0c;DHT11温湿度传感器&#xff0c;风扇&#xff…

Mac中配置Node.js前端vscode环境(第二期)

核心组件&#xff1a;vscode、谷歌浏览器、Node.js&#xff08;重点&#xff09;、git 一、Node.js安装&#xff08;nvm安装&#xff09; 点击macos中的终端&#xff0c;保持bash&#xff0c;而不是zsh 若为zsh&#xff0c;则可在终端中使用下面命令变成bash chsh -s /bin/…

基于SpringBoot和Thymeleaf的仿小米电商系统源码下载与安装指南-幽络源

项目概述 这是一个基于 Spring Boot 2.X 和 Thymeleaf 技术栈的仿小米电商系统。该项目包括了前台商城系统和后台管理系统&#xff0c;经幽络源测试具备完整的电商功能&#xff0c;适合用作学习、参考或作为开发电商系统的基础。 前台商城系统&#xff1a; 包括首页登录、商品…

Deepseek v3 的笔记

基本概述 Deepseek v3是Deepseek发布的旗舰模型&#xff0c;属于607B的混合专家&#xff08;MoE&#xff09;模型&#xff0c;其中活跃参数为37B。在当前的模型领域&#xff0c;它堪称最佳的开源模型&#xff0c;性能超越了Llama 3.1 405b、Qwen和Mistral等知名模型。根据基准…

机器学习中的常用特征选择方法及其应用案例

工业界中&#xff0c;特征选择是机器学习模型开发的重要步骤&#xff0c;可以提高模型性能、减少过拟合和加速计算。针对类别型特征和数值型特征&#xff0c;特征选择方法有所不同。以下详细讲解方法和案例。 一、类别型特征的特征选择 1. 基于卡方检验 卡方检验是一种衡量两…

Pyecharts SVG 标记使用笔记

Pyecharts SVG 标记使用笔记 在数据可视化中&#xff0c;图表的标记点常常用于突出显示重要数据点。Pyecharts 提供了丰富的标记点配置选项&#xff0c;其中使用 SVG 路径来自定义标记点的样式是一个非常强大的功能。本文将详细介绍如何在 Pyecharts 中使用 SVG 路径来定制标记…

Spring SpEL表达式由浅入深

标题 前言概述功能使用字面值对象属性和方法变量引用#this 和 #root变量获取类的类型调用对象(类)的方法调用类构造器类型转换运算符赋值运算符条件(关系)表达式三元表达式Elvis 操作符逻辑运算instanceof 和 正则表达式的匹配操作符 安全导航操作员数组集合(Array 、List、Map…

大型模型推理加速入门

一 推理加速基础 1. 了解基本概念与术语 首先需要了解关于大模型的一些基本的历史发展&#xff0c;概念和术语&#xff0c;不需要全部看懂&#xff0c;只需要掌握大概情况&#xff0c;可以通过下面这篇文章进行&#xff1a; 大模型入门基本概念&#xff0c;术语 2. 了解Tra…

仓库叉车高科技安全辅助设备——AI防碰撞系统N2024G-2

在当今这个高效运作、安全第一的物流时代&#xff0c;仓库作为供应链的中心地带&#xff0c;其安全与效率直接关系到企业的命脉。 随着科技的飞速发展&#xff0c;传统叉车作业模式正逐步向智能化、安全化转型&#xff0c;而在这场技术革新中&#xff0c;AI防碰撞系统N2024G-2…

python-redis访问指南

Redis&#xff08;Remote Dictionary Server&#xff09;是一种开源的内存数据结构存储&#xff0c;可用作数据库、缓存和消息代理。它功能强大且灵活&#xff0c;可根据需求调整架构和配置&#xff0c;以高性能、简单易用、支持多种数据结构而闻名&#xff0c;广泛应用于各种场…

Flash Attention V3使用

Flash Attention V3 概述 Flash Attention 是一种针对 Transformer 模型中注意力机制的优化实现&#xff0c;旨在提高计算效率和内存利用率。随着大模型的普及&#xff0c;Flash Attention V3 在 H100 GPU 上实现了显著的性能提升&#xff0c;相比于前一版本&#xff0c;V3 通…

UE5失真材质

渐变材质函数&#xff1a;RadialGradientExponential&#xff08;指数径向渐变&#xff09; 函数使用 UV 通道 0 来产生径向渐变&#xff0c;同时允许用户调整半径和中心点偏移。 用于控制渐变所在的位置及其涵盖 0-1 空间的程度。 基于 0-1 的渐变中心位置偏移。 源自中心的径…

Ansys Aqwa 中 Diffraction Analysis 的疲劳结果

了解如何执行疲劳分析&#xff0c;包括由 Ansys Aqwa 计算的海浪行为。 了解疲劳分析 大多数机器故障是由于负载随时间变化&#xff0c;而不是静态负载。这种失效通常发生在应力水平明显低于材料的屈服强度时。因此&#xff0c;当存在动态载荷时&#xff0c;仅依赖静态失效理…

MT8788安卓核心板_MTK8788核心板参数_联发科模块定制开发

MT8788安卓核心板是一款尺寸为52.5mm x 38.5mm x 2.95mm的高集成度电路板&#xff0c;专为各种智能设备应用而设计。该板卡整合了处理器、图形处理单元(GPU)、LPDDR3内存、eMMC存储及电源管理模块&#xff0c;具备出色的性能与低功耗特性。 这款核心板搭载了联发科的MT8788处理…

【UE5 C++课程系列笔记】19——通过GConfig读写.ini文件

步骤 1. 新建一个Actor类&#xff0c;这里命名为“INIActor” 2. 新建一个配置文件“Test.ini” 添加一个自定义配置项 3. 接下来我们在“INIActor”类中获取并修改“CustomInt”的值。这里定义一个方法“GetINIVariable” 方法实现如下&#xff0c;其中第16行代码用于构建配…