`
美丽的小岛
  • 浏览: 298392 次
  • 性别: Icon_minigender_1
  • 来自: 大连
社区版块
存档分类
最新评论

模运算<知识点>

阅读更多

模运算即求余运算。“模”是“Mod”的音译,模运算多应用于程序编写中。 Mod的含义为求余。模运算在数论和程序设计中都有着广泛的应用,从奇偶数的判别到素数的判别,从模幂运算到最大公约数的求法,从孙子问题到凯撒密码问题,无不充斥着模运算的身影。
  例如11 Mod 2,值为1
  上述模运算多用于程序编写,举一例来说明模运算的原理:
  Turbo Pascal对mod的解释是这样的:
  A Mod B=A-(A div B) * B (div含义为整除)

 

基本理论
  基本概念:
  给定一个正整数p,任意一个整数n,一定存在等式 n = kp + r ;
  其中k、r是整数,且 0 ≤ r < p,称呼k为n除以p的商,r为n除以p的余数。
  对于正整数p和整数a,b,定义如下运算:
  取模运算:a % p(或a mod p),表示a除以p的余数。
  模p加法:(a + b) % p ,其结果是a+b算术和除以p的余数,也就是说,(a+b) = kp +r,则(a + b) % p = r。
  模p减法:(a-b) % p ,其结果是a-b算术差除以p的余数。
  模p乘法:(a * b) % p,其结果是 a * b算术乘法除以p的余数。
  说明:
  1. 同余式:正整数a,b对p取模,它们的余数相同,记做 a ≡ b % p或者a ≡ b (mod p)。
  2. n % p得到结果的正负由被除数n决定,与p无关。例如:7%4 = 3, -7%4 = -3, 7%-4 = 3, -7%-4 = -3。
基本性质
  (1)若p|(a-b),则a≡b (% p)。例如 11 ≡ 4 (% 7), 18 ≡ 4(% 7)
  (2)(a % p)=(b % p)意味a≡b (% p)
  (3)对称性:a≡b (% p)等价于b≡a (% p)
  (4)传递性:若a≡b (% p)且b≡c (% p) ,则a≡c (% p)
运算规则
  模运算与基本四则运算有些相似,但是除法例外。其规则如下:
  (a + b) % p = (a % p + b % p) % p (1)
  (a - b) % p = (a % p - b % p) % p (2)
  (a * b) % p = (a % p * b % p) % p (3)
  (a^b) % p = ((a % p)^b) % p (4)
  结合率: ((a+b) % p + c) % p = (a + (b+c) % p) % p (5)
  ((a*b) % p * c)% p = (a * (b*c) % p) % p (6)
  交换率: (a + b) % p = (b+a) % p (7)
  (a * b) % p = (b * a) % p (8)
  分配率: ((a +b)% p * c) % p = ((a * c) % p + (b * c) % p) % p (9)
  重要定理:若a≡b (% p),则对于任意的c,都有(a + c) ≡ (b + c) (%p);(10)
  若a≡b (% p),则对于任意的c,都有(a * c) ≡ (b * c) (%p);(11)
  若a≡b (% p),c≡d (% p),则 (a + c) ≡ (b + d) (%p),(a - c) ≡ (b - d) (%p),
  (a * c) ≡ (b * d) (%p),(a / c) ≡ (b / d) (%p); (12)
  若a≡b (% p),则对于任意的c,都有ac≡ bc (%p); (13)
基本应用
  1.判别奇偶数
  奇偶数的判别是模运算最基本的应用,也非常简单。易知一个整数n对2取模,如果余数为0,则表示n为偶数,否则n为奇数。
  2.判别素数
  一个数,如果只有1和它本身两个因数,这样的数叫做质数(或素数)。例如 2,3,5,7 是质数,而 4,6,8,9 则不是,后者称为合成数或合数。
  判断某个自然数是否是素数最常用的方法就是试除法:用比该自然数的平方根小的正整数去除这个自然数,若该自然数能被整除,则说明其非素数。
  C++实现功能函数: 

 /* 
  函数名:IsPrime 
  函数功能:判别自然数n是否为素数。 
  输入值:int n,自然数n 
  返回值:bool,若自然数n是素数,返回true,否则返回false 
  */ 
  bool IsPrime(unsigned int n) 
  { 
  unsigned maxFactor = sqrt(n); //n的最大因子 
  for (unsigned int i=2; i<=maxFactor; i++) 
  { 
  if (n % i == 0) //n能被i整除,则说明n非素数 
  { 
  return false; 
  } 
  } 
  return true; 
  } 

  3. 最大公约数
  求最大公约数最常见的方法是欧几里德算法(又称辗转相除法),其计算原理依赖于定理:gcd(a,b) = gcd(b,a mod b)
  证明:a可以表示成a = kb + r,则r = a mod b
  假设d是a,b的一个公约数,则有d|a, d|b,而r = a - kb,因此d|r
  因此d是(b,a mod b)的公约数
  假设d 是(b,a mod b)的公约数,则d | b , d |r ,但是a = kb +r
  因此d也是(a,b)的公约数
  因此(a,b)和(b,a mod b)的公约数是一样的,其最大公约数也必然相等,得证。
  C++实现功能函数:

/* 
  函数功能:利用欧几里德算法,采用递归方式,求两个自然数的最大公约数 
  函数名:Gcd 
  输入值:unsigned int a,自然数a 
  unsigned int b,自然数b 
  返回值:unsigned int,两个自然数的最大公约数 
  */ 
  unsigned int Gcd(unsigned int a, unsigned int b) 
  { 
  if (b == 0) 
  return a; 
  return Gcd(b, a % b); 
  } 
  /* 
  函数功能:利用欧几里德算法,采用迭代方式,求两个自然数的最大公约数 函数名:Gcd 
  输入值:unsigned int a,自然数a 
  unsigned int b,自然数b 
  返回值:unsigned int,两个自然数的最大公约数 
  */ 
  unsigned int Gcd(unsigned int a, unsigned int b) 
  { 
  unsigned int temp; 
  while (b != 0) 
  { 
  temp = a % b; 
  a = b; 
  b = temp; 
  } 
  return a; 
  } 

  4.模幂运算
  利用模运算的运算规则,我们可以使某些计算得到简化。例如,我们想知道3333^5555的末位是什么。很明显不可能直接把3333^5555的结果计算出来,那样太大了。但我们想要确定的是3333^5555(%10),所以问题就简化了。
  根据运算规则(4)a^b% p = ((a % p)^b) % p ,我们知道3333^5555(%10)= 3^5555(%10)。由于3^4 = 81,所以3^4(%10)= 1。
  根据运算规则(3) (a * b) % p = (a % p * b % p) % p ,由于5555 = 4 * 1388 + 3,我们得到3^5555(%10)=(3^(4*1388) * 3^3)(%10)=((3^(4*1388)(%10)* 3^3(%10))(%10)
  =(1 * 7)(%10)= 7。
  计算完毕。
  利用这些规则我们可以有效地计算X^N(% P)。简单的算法是将result初始化为1,然后重复将result乘以X,每次乘法之后应用%运算符(这样使得result的值变小,以免溢出),执行N次相乘后,result就是我们要找的答案。
  这样对于较小的N值来说,实现是合理的,但是当N的值很大时,需要计算很长时间,是不切实际的。下面的结论可以得到一种更好的算法。
  如果N是偶数,那么X^N =(X*X)^[N/2];
  如果N是奇数,那么X^N = X*X^(N-1) = X *(X*X)^[N/2];
  其中[N]是指小于或等于N的最大整数。
  C++实现功能函数:

 /* 
  函数功能:利用模运算规则,采用递归方式,计算X^N(% P) 
  函数名:PowerMod 
  输入值:unsigned int x,底数x 
  unsigned int n,指数n 
  unsigned int p,模p 
  返回值:unsigned int,X^N(% P)的结果 
  */ 
  unsigned int PowerMod(unsigned int x, unsigned int n, unsigned int p) 
  { 
  if (n == 0) 
  { 
  return 1; 
  } 
  unsigned int temp = PowerMod((x * x)%p, n/2, p); //递归计算(X*X)^[N/2] 
  if ((n & 1) != 0) //判断n的奇偶性 
  { 
  temp = (temp * x) % p; 
  } 
  return temp; 
  } 

 5.《孙子问题(中国剩余定理)》
  在我国古代算书《孙子算经》中有这样一个问题:
  “今有物不知其数,三三数之剩二,五五数之剩三,七七数之剩二,问物几何?”意思是,“一个数除以3余2,除以5余3,除以7余2.求适合这个条件的最小数。”
  这个问题称为“孙子问题”.关于孙子问题的一般解法,国际上称为“中国剩余定理”.
  我国古代学者早就研究过这个问题。例如我国明朝数学家程大位在他著的《算法统宗》(1593年)中就用四句很通俗的口诀暗示了此题的解法:
  三人同行七十稀,五树梅花甘一枝,七子团圆正半月,除百零五便得知。
  "正半月"暗指15。"除百零五"的原意是,当所得的数比105大时,就105、105地往下减,使之小于105;这相当于用105去除,求出余数。
  这四句口诀暗示的意思是:当除数分别是3、5、7时,用70乘以用3除的余数,用21乘以用5除的余数,用15乘以用7除的余数,然后把这三个乘积相加。加得的结果如果比105大,就除以105,所得的余数就是满足题目要求的最小正整数解。
  根据剩余定理,我把此种解法推广到有n(n为自然数)个除数对应n个余数,求最小被除数的情况。输入n个除数(除数不能互相整除)和对应的余数,计算机将输出最小被除数。
  C++实现功能函数:  

 /* 
  函数名:ResidueTheorem 
  函数功能:运用剩余定理,解决推广了的孙子问题。通过给定n个除数(除数不能互相整除)和对应的余数,返回最小被除数 
  输入值:unsigned int devisor[],存储了n个除数的数组 
  unsigned int remainder[],存储了n个余数的数组 
  int length,数组的长度 
  返回值:unsigned int, 最小被除数 
  */ 
  unsigned int ResidueTheorem(const unsigned int devisor[], const unsigned int remainder[], int length) 
  { 
  unsigned int product = 1; //所有除数之乘积 
  for (int i=0; i<length; i++)//计算所有除数之乘积 
  { 
  product *= devisor[i]; 
  } 
  //公倍数数组,表示除该元素(除数)之外其他除数的公倍数 
  unsigned int *commonMultiple = new unsigned int(length); 
  for (int i=0; i<length; i++)//计算除该元素(除数)之外其他除数的公倍数 
  { 
  commonMultiple[i] = product / devisor[i]; 
  } 
  unsigned int dividend = 0; //被除数,就是函数要返回的值 
  for (int i=0; i<length; i++)//计算被除数,但此时得到的不是最小被除数 
  { 
  unsigned int tempMul = commonMultiple[i]; 
  //按照剩余理论计算合适的公倍数,使得tempMul % devisor[i] == 1 
  while (tempMul % devisor[i] != 1) 
  { 
  tempMul += commonMultiple[i]; 
  } 
  dividend += tempMul * remainder[i]; //用本除数得到的余数乘以其他除数的公倍数 
  } 
  delete []commonMultiple; 
  return (dividend % product); //返回最小被除数 
  } 

  6. 凯撒密码
  凯撒密码(caeser)是罗马扩张时期朱利斯o凯撒(Julius Caesar)创造的,用于加密通过信使传递的作战命令。
  它将字母表中的字母移动一定位置而实现加密。注意26个字母循环使用,z的后面可以看成是a。
  例如,当密匙为k = 3,即向后移动3位时,若明文为”How are you!”,则密文为”Krz duh btx!”。
  凯撒密码的加密算法极其简单。其加密过程如下:
  在这里,我们做此约定:明文记为m,密文记为c,加密变换记为E(key1,m)(其中key1为密钥),
  解密变换记为D(key2,m)(key2为解密密钥)(在这里key1=key2,不妨记为key)。
  凯撒密码的加密过程可记为如下一个变换:c≡m+key (mod n) (其中n为基本字符个数)
  同样,解密过程可表示为:m≡c+key (mod n) (其中n为基本字符个数)
  C++实现功能函数:   

/* 
  函数功能:使用凯撒密码原理,对明文进行加密,返回密文 函数名:Encrypt 
  输入值:const char proclaimedInWriting[],存储了明文的字符串 
  char cryptograph[],用来存储密文的字符串 
  int keyey,加密密匙,正数表示后移,负数表示前移 
  返回值:无返回值,但是要将新的密文字符串返回 
  */ 
  void Encrypt(const char proclaimedInWriting[], char cryptograph[], int key) 
  { 
  const int NUM = 26; //字母个数 
  int len = strlen(proclaimedInWriting); 
  for (int i=0; i<len; i++) 
  { 
  if (proclaimedInWriting[i] >= 'a' && proclaimedInWriting[i] <= 'z') 
  {//明码是大写字母,则密码也为大写字母 
  cryptograph[i] = (proclaimedInWriting[i] - 'a' + key) % NUM + 'a'; 
  } 
  else if (proclaimedInWriting[i] >= 'A' && proclaimedInWriting[i] <= 'Z') 
  {//明码是小写字母,则密码也为小写字母 
  cryptograph[i] = (proclaimedInWriting[i] - 'A' + key) % NUM + 'A'; 
  } 
  else 
  {//明码不是字母,则密码与明码相同 
  cryptograph[i] = proclaimedInWriting[i]; 
  } 
  } 
  cryptograph[len] = '\0'; 
  } 
  /* 
  函数功能:使用凯撒密码原理,对密文进行解密,返回明文 函数名:Decode 
  输入值:char proclaimedInWriting[],用来存储明文的字符串 
  const char cryptograph[],存储了密文的字符串 
  int keyey,解密密匙,正数表示前移,负数表示后移(与加密相反) 
  返回值:无返回值,但是要将新的明文字符串返回 
  */ 
  void Decode(const char cryptograph[], char proclaimedInWriting[], int key) 
  { 
  const int NUM = 26; //字母个数 
  int len = strlen(cryptograph); 
  for (int i=0; i<len; i++) 
  { 
  if (cryptograph[i] >= 'a' && cryptograph[i] <= 'z') 
  {//密码是大写字母,则明码也为大写字母,为防止出现负数,转换时要加个NUM 
  proclaimedInWriting[i] = (cryptograph[i] - 'a' - key + NUM) % NUM + 'a'; 
  } 
  else if (cryptograph[i] >= 'A' && cryptograph[i] <= 'Z') 
  {//密码是小写字母,则明码也为小写字母 
  proclaimedInWriting[i] = (cryptograph[i] - 'A' - key + NUM) % NUM + 'A'; 
  } 
  else 
  {//密码不是字母,则明码与明密相同 
  proclaimedInWriting[i] = cryptograph[i]; 
  } 
  } 
  proclaimedInWriting[len] = '\0'; 
  } 

   模运算及其简单应用就先讲到这了,其实模运算在数学及计算机领域的应用非常广泛,我这这里搜集整理了一些最最基本的情形,希望能够起到一个抛砖引玉的作用,让更多的人关注模运算,并及其应用到更广阔的领域中。

 

 http://baike.baidu.com/view/2385246.htm#ref_[1]_2385246

分享到:
评论

相关推荐

    数据结构(C++)有关练习题

    &lt;br&gt;&lt;br&gt;实验二 单链表结构及计算&lt;br&gt;实验目的:&lt;br&gt;通过实验掌握下列知识: &lt;br&gt;1、熟悉线性表的基本运算在两种存储结构(顺序结构和链式结构)上的实现;&lt;br&gt;2、继续熟悉VC编程、编译和调试环境;&lt;br&gt;内容及步骤:&lt;br...

    Java基础知识点总结.docx

    Java学习更是如此,知识点总结目录如下: 目录 一、 Java概述 3 二、 Java语法基础 5 数据类型 5 运算符号 14 语句 15 函数 15 方法重载(Overloadjing)与重写(Overriding) 16 数组 17 总结 18 三、 常见关键字 ...

    PHP基础教程 是一个比较有价值的PHP新手教程!

    &lt;/TITLE&gt; &lt;/HEAD&gt; &lt;BODY&gt; &lt;H1&gt; First PHP page &lt;/H1&gt; &lt;HR&gt; &lt;? // Single line C++ style comment /* printing the message */ echo "Hello World!"; # Unix style single line comment ?&gt; &lt;/BODY&gt; &lt;/HTML&gt; 2.4 数据...

    十天学会ASP.net--我认为ASP.NET比ASP难很多,希望大家做好准备

    在这里我要说明两点:1、我的示例文件总是有A和B分别是用C#和VB写的,演示的图片就用C#那一种的,都一样嘛,教程里面代码也是写两种用&lt;hr&gt;分割开,大家可以比较一下。2、我写教程的时候用的都是记事本来编写APS.NET...

    自考计算机应用基础知识点-范本模板.docx

    自考计算机应用基础知识点-范本模板 自考计算机应用基础知识点-范本模板全文共16页,当前为第1页。自考计算机应用基础知识点-范本模板全文共16页,当前为第1页。计算机基础知识点 自考计算机应用基础知识点-范本模板...

    软考系分之数据库E-R模型、关系模型等以及关系运算

    内容概要:关系数据库的关系运算,包括交、并、差、笛卡尔积、自然连接、选择...适用人群:系分备考者、产品经历、软件开发等,也适用于有相关知识点学习兴趣的小伙伴;其他:思维导图的方式介绍知识点,标注重点和示例

    《数据库系统概论》知识点总结.docx

    《数据库系统概论》知识点总结全文共10页,当前为第1页。《数据库系统概论》知识点总结全文共10页,当前为第1页。一、选择题: 《数据库系统概论》知识点总结全文共10页,当前为第1页。 《数据库系统概论》知识点...

    AI学习知识点.xmind

    *AI学习知识点* 1. 基础知识 概率论 微积分与逼近论 极限、微分、积分的基本概念 利用逼近的思想理解微积分,利用积分的方式理解概率论 概率论的基础 古典模型 常见的概率分布 大数定理和中心极限定理 ...

    matlab常用知识点总结.rar

    各章节的主要知识点如下: 第一章:安装及使用前的准备 1 了解MATLAB的基本发展历史; 2 掌握MATLAB的启动、熟悉其桌面平台; 3 掌握MATLAB的帮助系统的使用。 第二章: 数值计算功能 1 熟悉MATLAB中的...

    C#微软培训资料

    &lt;&lt;page 1&gt;&gt; page begin==================== 目 目目 目 录 录录 录 第一部分 C#语言概述.4 第一章 第一章第一章 第一章 .NET 编 编 编程语言 程语言编程语言 程语言 C#.4 1.1 Microsoft...

    Grasshopper运算器教程V1.0.pdf

    不同于Rhino Scrip,Grasshopper不需要太多任何的程序语言的知识就可以通过一些简单的流程方法达到设计师所想要的模型. 个人认为,Grasshopper其很大的价值在于它是以自己独特的方式完整记录起始模型(一个点或一个...

    数据库原理概论理论知识点总结.pdf

    数据库原理概论理论知识点总结 数据库概论 ⼀.绪论 1.1 数据库系统概述 数据是描述事物的符号记录,是数据库中存储的基本对象。 数据的解释是指对数据含义的说明,数据的含义称为数据的语义,数据与其语义是不可分割...

    数据库系统工程师知识点总结(共144页).docx

    也称算术逻辑单元,对数据进行算术运算和逻辑运算 加法器(累加器): 专门存放算术或逻辑运算的操作数和运算结果的寄存器。  程序状态寄存器 PSW: 用来存放两类信息:一类是体现当前指令执行结果的各种状态信息...

    汇编语言(各章知识点)

    汇编语言(各章知识点) 课程介绍 第1章 预备知识  1.1 汇编语言的由来及其特点  1 机器语言  2 汇编语言  3 汇编程序  4 汇编语言的主要特点  5 汇编语言的使用领域  1.2 数据的表示和类型  1 数值数据...

    计算机科学导论1.docx

    配置语言、操作系统、外部设备、运算速度 纠错 得分: 4 知识点: 5.1 计算机硬件(三大子系统) 收起解析 答案 B 解析 2. (4分)在计算机中,( )子系统存储数据和程序。 A. 算术逻辑单元 B. 输入/输出 C. 存储器 ...

    历届系统分析师考试数学知识点细分统计

    年份 试题号 知识点 2004 上 60 单射函数概念 2000 15 概率的性质,泊松分布概念 1999 15 简单概率,泊松分布 2001 70~75 概率分布、数学期望、方差 2002 57~65 随机变量正态分布 2004上 61 全概率公式应用 1993 ...

    架构师考试-快速通关-知识点

    关系运算-比较 管理信息系统规划的方法 架构描述语言-ADL 架构权衡分析方法 路由-层次化 论文-结尾/软件可靠性/设计模式/数据访问层设计/微服务/摘要/质量属性和评估方法 评价标准-web服务器 评价标准-操作系统、...

    Access2010数据库基础教程教案.doc

    " "课时安排 " "45分钟讲授知识点,45分钟上机练习 " "教学活动的组织形式 " "讲练结合 " "教学手段(教学媒体的运用) " "多媒体交互系统 " "教学过程的设计 " "介绍该门课程的特点及要求,10分钟; " "介绍数据库...

    数据库系统基础知识.pdf

    Visual FoxPuo 程序设计基础 第1章 数据库系统基础知识 数据库基础知识 1.1 数据模型 1.2 数据库系统 1.3 关系数据库 1.4 数据库系统的体系结构与开发工具 1.5 1.1 数据库基础知识 通常,把用计算机对数据进行处理的...

Global site tag (gtag.js) - Google Analytics