首页 > 其他分享 >数字黑洞

数字黑洞

时间:2022-11-22 19:13:09浏览次数:63  
标签:数字 min int max 黑洞 循环 位数

      一、4位数的数字黑洞

       任意给定一个4位数(不能所有位都相同),比如:3278,重新组合出最大数:8732,再重新组合出最小数:2378,相减,得到新的4位数(如不足则补0),重复这个过程,最后必然得到一个数字:6174。这个现象被称为:数字黑洞。

       编写一个程序,验证这个现象。

例如,给出四位数3278,则有验证过程如下:

8732-2378=6354

6543-3456=3087

8730-378=8352

8532-2358=6174

再如,给出四位数1011,则有验证过程如下:

1110-111=999  (不足4位,补0)

9990-999=8991 

9981-1899=8082

8820-288=8532

8532-2358=6174

        (1)编程思路。

编写函数void parse(int n,int *max,int *min),该函数的功能求出由整数n的四位数字组合成的最大数和最小数,分别通过形参max和min返回。

在函数中,定义一个数组int a[4],用于保存整数n的4位数字,然后将数组a按从小到大的顺序排列,之后各数字顺序组成最小数,逆序组成最大数。

(2)源程序。

#include <stdio.h>
void parse(int n,int *max,int *min)
{
    int a[4],i,j,t;
    for (i=0; i<4; i++)
    {
        a[3-i] = n%10;
        n/=10;
    }
    for (i=0; i<3; i++)    // 将四个数字按从小到大顺序排列
        for (j=0; j<3-i; j++)
            if (a[j]>a[j+1])
            {
                t = a[j+1];
                a[j+1] = a[j];
                a[j] = t;
            }
    t=0;
    for (i=0; i<4; i++)
        t = t * 10 + a[i] ;
    *min=t;
    t = 0;
    for (i=3; i>=0; i--)
        t= t * 10 + a[i];
    *max=t;
}
int main()
{
    int n,max,min;
    scanf("%d",&n);
    do
    {
       parse(n,&max,&min);
       printf("%d-%d=%d\n",max,min,max-min);
       n=max-min;
    }while (n!=6174);
    return 0;
}

        二、5位数的数字黑洞

        任意一个5位数,比如:12345,把它的各位数字打乱,重新排列,可以得到一个最大的数:54321,一个最小的数12345。求这两个数字的差,得:41976,把这个数字再次重复上述过程(如果不足5位,则前边补0)。如此往复,数字会落入某个循环圈(称为数字黑洞)。

例如,刚才的数12345会落入:[82962, 75933, 63954, 61974] 这个循环圈。计算过程如下:

54321 - 12345 = 41976

97641 - 14679 = 82962

98622 - 22689 = 75933

97533 - 33579 = 63954

96543 - 34569 = 61974

97641 - 14679 = 82962

[82962,75933,63954,61974] 

再如,整数11211会落入:[74943,62964,71973,83952]这个循环圈。计算过程如下:

21111 - 11112 = 9999   (不足5位,则前边补0)

99990 - 9999 = 89991

99981 - 18999 = 80982

98820 - 2889 = 95931

99531 - 13599 = 85932

98532 - 23589 = 74943

97443 - 34479 = 62964

96642 - 24669 = 71973

97731 - 13779 = 83952

98532 - 23589 = 74943

[74943,62964,71973,83952]

还如,整数50000会落入:[53955,59994]这个循环圈。计算过程如下:

50000 - 5 = 49995

99954 - 45999 = 53955

95553 - 35559 = 59994

99954 - 45999 = 53955

[53955,59994]

         编写一个程序,找到5位数所有可能的循环圈,并输出,每个循环圈占1行。其中5位数全都相同则循环圈为 [0],这个可以不考虑。循环圈的输出格式仿照:

[82962, 75933, 63954, 61974]   其中数字的先后顺序可以不考虑。

         (1)编程思路。

 如同上面4位数的数字黑洞,编写函数int next(int n) ,其功能是求5位整数n的各位数字所组成的最大数与最小数的差值,并将求得的差值作为函数值返回。

为了找出一个整数n的循环圈,编写一个函数void heidong(int n),寻找整数n的循环圈。在函数中,定义一个数组int a[20]用于保存计算过程中的每一个差值。

初始时,置a[0]=n,之后用next()函数求得n的各位数字组成的最大数与最小数的差值,保存到a[1]中,即a[1]=next(a[0]),再求 a[2]=next(a[1]),…,a[i]=next(a[i-1])。

每次求得了a[i]后,将a[i]与a[0]~a[i-1]中保存的各数依次比较,若每个a[j]==a[i](0≤j≤i-1),则找到了循环圈 a[j]~a[i-1]。输出这个循环圈。并将这个循环圈中的各个数保存到全局数组b中。

       之所以要将循环圈中的数保存到全局数组中,是因为很多5位数会落在同一个循环圈中。例如,整数10000会落在[74943,62964,71973,83952]循环圈中,11112、11121、11211、…这些数同样落在这个循环圈中。这样当计算10000找到了循环圈后,将74943、62964、71973、83952这4个数保存在全局数组b中,当以后计算到11112时,找到了循环圈,由于循环圈中的数74943已在全局数组b中存在,因此这个循环圈是重复的,无需输出。这样,可以用循环

        for(i=10000;i<99999;i++)
        {
                 if (next(i)==0) continue; // 各位数字全部相同,忽略
                 heidong(i);
        }

找出所有5位数可能的循环圈。

(2)源程序。

#include<stdio.h>
int b[12]={0}, cnt=0; // 保存各循环圈中的数避免重复,cnt为黑洞中数的个数
int next(int n)       // 整数n各位数字组成的最大数与最小数的差
{
    int a[5],i,j;
    for (i=0;i<5;i++)
    {
        a[i]=n%10;
        n/=10;
    }
    for (i=0;i<4;i++)
        for (j=0;j<4-i;j++)
           if (a[j]>a[j+1])
           {
              int t;
              t=a[j]; a[j]=a[j+1]; a[j+1]=t;
           }
    int max=0,min=0;
    for (i=0;i<5;i++)
    {
        min=min*10+a[i];
        max=max*10+a[4-i];
    }
    return max-min;
}
void heidong(int n)
{
    int a[20],flag=0;
    a[0]=n;
    int i,j,k;
    for (i=1; ;i++)
    {
        a[i]=next(a[i-1]);
        for (j=0;j<i;j++)
        {
            if(a[i]==a[j])          // a[j]~a[i-1]之间的数构成循环圈
            {
                for (k=0;k<cnt;k++) // 看当前循环圈中的数是否保存过,本质是查重
                {
                    if(a[i]==b[k]) { flag=1; break;}
                }
                if (flag!=1)      // 输出循环圈中的各数,同时保存到全局数组b中
                {
                    printf("[%d",a[j]);
                    b[cnt++]=a[j];
                    for (k=j+1;k<i;k++)
                    {
                       printf(",%d",a[k]);
                       b[cnt++]=a[k];
                    }
                    printf("]\n");
                    flag=1;
                }
            }
        }
        if (flag==1) break;
    }
}
int main()
{
    int i;
    for(i=10000;i<99999;i++)
    {
       if (next(i)==0) continue;  // 各位数字全部相同,忽略
       heidong(i);
    }
    return 0;
}

        运行程序可知,所有5位数可能的循环圈有3个,如下。

[74943,62964,71973,83952]
[63954,61974,82962,75933]
[53955,59994]

标签:数字,min,int,max,黑洞,循环,位数
From: https://www.cnblogs.com/cs-whut/p/16914042.html

相关文章

  • 878. 第 N 个神奇数字
    878.第N个神奇数字题解:二分+gcd1~n中,能被a整除的个数n/a,能被b整除的个数n/b,能被a且b整除的个数n/gcd(a,b)则1~n中能被a或b整除的个数为n/a+n/b......
  • 输入四个数字判断最大数和最小数
    这个问题很简单我们直接看代码有好几种解决方式但我的实力实在有限暂时只会这两种请看我博客的大神们见谅#include<stdio.h>intmain(){inta,b,c,d,max=0,min=0;//......
  • 数字化时代,学校VR实训教学解决方案开启VR教学新纪元
    近年来国家大力发展职业教育改革,教育部等十四部门研究制定了《职业院校全面开展职业培训促进就业创业行动计划》,计划中明确指出:职业院校要积极开发VR(虚拟现实技术)等数字化......
  • 广州华锐互动|构建智慧城市数字化三维实景建模平台
    在技术日新月异的时代,智慧城市建设发展也在不断推进,从物理世界到虚拟世界,人们对于未来城市的构想正在智慧城市中逐步实现。智慧城市的构建离不开物联网、人工智能、虚拟现......
  • 878. 第 N 个神奇数字
    第N个神奇数字一个正整数如果能被a或b整除,那么它是神奇的。给定三个整数n, a,b,返回第n个神奇的数字。因为答案可能很大,所以返回答案 对 109 +7取模......
  • 如何用App实现巡检业务数字化?以YonBuilder移动开发平台APICloud为例
    巡检是企事业单位的常见场景之一,以消防检查为例,秋冬季节气温下降、生产繁忙,用火、用电、用气情况大量增加,消防安全事件多发,一款消防检查app可以有效减少繁复工作、提升巡......
  • 数字先锋 | 随时随地云端阅片,“云胶片”时代来啦!
    作为现代医疗必不可少的诊断方法,医学影像数据在医疗数据中的占比高达90%且正以每年30%的速度递增,而影像医生就业人数年增长率仅4%。这意味着,全国总人数不到20万的放射科医生......
  • “元宇宙家园”国脉大厦展馆上线 天翼云实时云渲染筑基未来数字世界
    11月12日至13日,2022世界VR产业大会在南昌举办。在此期间,以“共建元宇宙生态,点亮新数智未来”为主题的中国电信生态论坛召开,中国电信天翼云携手新国脉数字文化股份有限公司(简......
  • 【华为OJ11】数字颠倒
    题目描述描述:输入一个整数,将这个整数以字符串的形式逆序输出程序不考虑负数的情况,若数字含有0,则逆序形式也含有0,如输入为100,则输出为001输入描述:输入一个int整数......
  • 【数组11】和为S的两个数字
    题目描述输入一个递增排序的数组和一个数字S,在数组中查找两个数,使得他们的和正好是S,如果有多对数字的和等于S,输出两个数的乘积最小的。 输出描述:对应每个测试案例......