【基础算法】几种特殊数(素数、公约数、完全数、亲密数) & C++实现

简介: 素数又称为质数,它指在一个大于1的自然数中,除了1和它自身外,没法被其他自然数整除的数。比1大,但不是素数的数称为合数。0和1既不是素数,也不是合数。因为素数的分布没有明显的规律,所以在程序中一般根据素数的定义来判断该数是否为素数。例如哥德巴赫猜想:哥德巴赫通过大量的数据猜测,所有不小于6的偶数,都可以表示为两个奇素数之和。后人将其称之为“1+1”。并且,对于每个不小于9的奇数,都可以表示为三个奇素数之和。

●素数


       素数又称为质数,它指在一个大于1的自然数中,除了1和它自身外,没法被其他自然数整除的数。比1大,但不是素数的数称为合数。0和1既不是素数,也不是合数。因为素数的分布没有明显的规律,所以在程序中一般根据素数的定义来判断该数是否为素数。例如哥德巴赫猜想:哥德巴赫通过大量的数据猜测,所有不小于6的偶数,都可以表示为两个奇素数之和。后人将其称之为“1+1”。并且,对于每个不小于9的奇数,都可以表示为三个奇素数之和。

#include<iostream>
using namespace std;
class isprime {
public:
  void isprime1()
  {
  int k=0;
  for (int i = a; i <= b; i++)
  {
    if(i==2||i==3)
    { 
    k = 1;
    }
    for (int j = 2; j <= i/2 ; j++)   
    {
    if (i % j == 0)
    {
      k = 0;
      break;
    }
    else
    {
      k = 1;
    }
    }
    if (k == 1)
    {
    cout << i << " ";
    }
  }
  }
  int a;
  int b;
};
void text()
{
  isprime ip;
  cout << "请输入一个区间范围:" << endl;
  cin >> ip.a >> ip.b;
  cout << "该区间范围内的素数有:" << endl;
  ip.isprime1();
}
int main()
{
  text();
}

43f1bdfd245a8f76b3aef1218202f718_f9c1250ce3804bb48a84dd729474b2bd.png


●最大公约数


       最大公约数和最小公倍数是我们频繁接触的一对数。如果有一个自然数m能被自然数n整除,则称m为n的倍数,b为a的约数。多个自然数公共的约数,叫做这几个自然数的公约数。而公约数中最大的一个,就称为这几个自然数的最大公约数。


        计算两个自然数最大公约数的传统算法就是欧几里得的辗转相除法,对于要对多个自然数寻找其最大公约数可以通过多次辗转相除法得到,具体步骤如下:


①已知两自然数m、n,假设m>n;


②计算m除以n,将得到的余数记为r;


③如果r=0,则n为求得的最大公约数,否则执行下面一步;


④将n的值保存到m中,将r的值保存到n中,重复执行步骤②和③,直到r=0,便得到最大公约数;

#include<iostream>
using namespace std;
class max_public_number {
public:
  int max_public_number1()
  {
  int temp;
  if (m > n){
    m = m;
    n = n;
  }
  else{
    temp = m;
    m = n;
    n = temp;
  }
  r = m % n;
  while (r!= 0)   //辗转相除法
  {
    m = n;
    n = r;
    r = m % n;
  }
  return n;
  }
  void showresult()
  {
  cout << n << endl;
  }
  int m;
  int n;
  int r;
};
void text()
{
  max_public_number mpn;
  cout << "请输入两个数:" << endl;
  cin >> mpn.m>>mpn.n;
  cout << "这两个数的最大公约数为:" << endl;
  mpn.max_public_number1();
  mpn.showresult();
}
int main()
{
  text();
}

81115f69e51b8c8807ab842d6a0c20d5_3458559af9ba4fdab903e170b2444f98.png

●完全数


       当一个自然数的所有真因子之和等于该自然数,那么该自然数便是完全数(完全数的尾数都是6或8)。


1fe15f689371542d78855d465fd98a97_c6ac388390a44344b4b08a496286952a.png


完全数的特殊性质:


1.每个完全数都可以表示成连续自然数的和

cee8d9ae3ce8a7fbc29ad3039cf28a4e_c5bf684fffc04fc7b22b26255e456ffb.png

2. 每个完全数都是调和级数

c62cb2027faf4f339e0ef08dcd2b59f4_363320a7a16c4f1ab7614ded5829d6f9.png

3.每个完全数都可以表示为2的一些连续正整数次幂之和

ac64ba6aa6bb884f22ac310a2d94d3b9_40e50f5d1bee4e7eadd3859da635720c.png

4.除6之外的完全数都可以表示成连续的奇立方之和

95591201f37271cd6b65850543a36de9_8eb27cf73e83408189f13f319e402997.png


#include<iostream>
using namespace std;
class perfectnum {
public:
  void perfectnum1()
  {
  for (int i = 1; i < range; i++)
  {
    long num = i;
    long sum = num;   //这里将num赋给sum,在下面num去判断寻找因子,sum去进行逐渐判断是否为完全数
    long count = 0;  //记数组下标数
    for (int j = 1; j < num; j++)
    {
    if (num % j == 0)   //判断j是否为num的真因子
    {
      p[count++] = j;  //若是因子,我们将其记录在数组内
      sum -= j;  //对sum进行逐减因子
    }
    }
    if (sum == 0) //如果sum能被因子逐减为0,则说明他为完全数
    {
    cout << "完全数:" << num << endl;
    for(int k=0;k<count;k++)
    { 
      cout << p[k] << " ";    //输出每个完全数的因子
    }
    cout << endl;
    }
  }
  }
  long p[100];
  int range;
};
void text()
{
  perfectnum pn;
  cout << "请输入一个范围:" << endl;
  cin >> pn.range;
  cout << "该范围内的完全数为:" << endl;
  pn.perfectnum1();
}
int main()
{
  text();
}

d1fd2368740a0a5f47dafd9de45c0c0d_251e21452b8344f09bc34394120497b5.png


●亲密数


       如果整数a的因子和等于整数b,整数b的因子和等于整数a,因子包括1但不包括本身,且a不等于b,则称a,b为亲密数对。


比如,220的因子为1、2、4、5、10、11、20、22、44、55、110;284的因子为1、2、4、71、142。而220的因子和1+2+4+5+10+11+20+22+44+55+110等于284;284的因子和1+2+4+71+142等于220,并且因子包括1但不包括本身,220不等于284,所以这两数为亲密数对。

#include<iostream>
using namespace std;
class friendnum {
public:
  void friendnum1(int i)
  {
    for(int size=0;size<100;size++)  //每次的调用需将数组重复置空
    { 
    a[size] = b[size] = 0;
    }
    count = 0;  //记录数组下标
    int sum_1 = 0;
    for (int j = 1; j < i / 2 + 1; j++)   //求数i的因子
    {
    if (i % j == 0)   //i能被j整除,j为因子
    {
      a[count++] = j;  //保存因子到数组a中
      sum_1 += j;  //累加因子之和
    }
    }
    count = 0;  //置零,重新记录数组下标
    int sum_2 = 0;
    for (int k = 1; k < sum_1 / 2 + 1; k++)   //将数i因子之和sum_1进行因子分解
    {
    if (sum_1 % k == 0)   //sum_1能被k整除
    {
      b[count++] = k;   //保存因子到数组b中
      sum_2 += k;  //累加因子之和
    }
    }
    if (sum_2 == i && i < sum_1)
    {
    cout<<sum_2<<"与"<<sum_1<< "是亲密数," <<"并前两数的因子如下:"<< endl;
    count = 0;
    while (a[count]!=0)
    {
      cout << a[count] <<" ";
      count++;
    }
    cout << endl;
    count = 0;
    while(b[count]!=0)
    { 
      cout << b[count] << " ";
      count++;
    }
    cout << endl;
    }
  }
  int a[100];
  int b[100];
  int range;
  int count;
};
void text()
{
  friendnum fn;
  cout << "请输入一个范围:" << endl;
  cin >> fn.range;
  cout << "该范围内的亲密数为:" << endl;
  for (int a = 1; a <= fn.range; a++)
  {
  fn.friendnum1(a);
  }
}
int main()
{
  text();
}

1aa82ac926006d0107b36a225ef52672_b32ebc9e1deb45db9c72cdc3965e1e51.png


目录
相关文章
|
3月前
|
算法 测试技术 C++
【动态规划算法】蓝桥杯填充问题(C/C++)
【动态规划算法】蓝桥杯填充问题(C/C++)
|
2天前
|
存储 算法 测试技术
【C++数据结构——树】二叉树的遍历算法(头歌教学实验平台习题) 【合集】
本任务旨在实现二叉树的遍历,包括先序、中序、后序和层次遍历。首先介绍了二叉树的基本概念与结构定义,并通过C++代码示例展示了如何定义二叉树节点及构建二叉树。接着详细讲解了四种遍历方法的递归实现逻辑,以及层次遍历中队列的应用。最后提供了测试用例和预期输出,确保代码正确性。通过这些内容,帮助读者理解并掌握二叉树遍历的核心思想与实现技巧。
16 2
|
11天前
|
存储 算法 安全
基于红黑树的局域网上网行为控制C++ 算法解析
在当今网络环境中,局域网上网行为控制对企业和学校至关重要。本文探讨了一种基于红黑树数据结构的高效算法,用于管理用户的上网行为,如IP地址、上网时长、访问网站类别和流量使用情况。通过红黑树的自平衡特性,确保了高效的查找、插入和删除操作。文中提供了C++代码示例,展示了如何实现该算法,并强调其在网络管理中的应用价值。
|
8天前
|
存储 算法 安全
基于哈希表的文件共享平台 C++ 算法实现与分析
在数字化时代,文件共享平台不可或缺。本文探讨哈希表在文件共享中的应用,包括原理、优势及C++实现。哈希表通过键值对快速访问文件元数据(如文件名、大小、位置等),查找时间复杂度为O(1),显著提升查找速度和用户体验。代码示例展示了文件上传和搜索功能,实际应用中需解决哈希冲突、动态扩容和线程安全等问题,以优化性能。
|
3月前
|
存储 算法 C++
高精度算法(加、减、乘、除,使用c++实现)
高精度算法(加、减、乘、除,使用c++实现)
851 0
高精度算法(加、减、乘、除,使用c++实现)
|
3月前
|
存储 算法 决策智能
【算法】博弈论(C/C++)
【算法】博弈论(C/C++)
|
3月前
|
存储 算法 C++
【算法】哈希映射(C/C++)
【算法】哈希映射(C/C++)
|
3月前
|
机器学习/深度学习 人工智能 算法
【算法】最长公共子序列(C/C++)
【算法】最长公共子序列(C/C++)
|
3月前
|
人工智能 算法 BI
一篇带你速通差分算法(C/C++)
一篇带你速通差分算法(C/C++)
|
3月前
|
人工智能 算法 C++
一篇带你速通前缀和算法(C/C++)
一篇带你速通前缀和算法(C/C++)