ZOJ 2529 - A+B in Hogwarts 解题报告

简介: 题目2529:A+B in Hogwarts           链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=1535           题目描述:哈利波特去的魔法学院使用一种特殊进制法表示数字:第i位用第i个素数为进制(radix),例如“个位”的进制为第一个素数2,“十位”的进制为第二个素数3,“百位”的进制为第三个素数5,...依此类推。

          题目2529:A+B in Hogwarts

          链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=1535

          题目描述:哈利波特去的魔法学院使用一种特殊进制法表示数字:第i位用第i个素数为进制(radix),例如“个位”的进制为第一个素数2,“十位”的进制为第二个素数3,“百位”的进制为第三个素数5,...依此类推。例如,十进制数2表示为10,十进制数6被表示为100。现在要求你为哈利波特写一个简单的计算器用于计算a+b的结果,每一行输入魔法学院的数字a和b,两个数字用空格分割,数字的每一位用逗号分割,要求计算出a+b的结果,并用魔法学院的表示法表示(注意,输入数字最多不超过25位)。

          示范输入:

          1,0 2,1
          4,2,0 1,2,0
          1 10,6,4,2,1

          示范输出:

          1,0,1
          1,1,1,0
          1,0,0,0,0,0

          该题目的难度不大,但对输入的解析和正确输出上有一定的技巧性,因此不能算特别简单的题。看到该题目的第一印象,我想把输入的数字a和b都转换为常规的int整数,然后再把a+b的结果转换为该特殊进制,但是这个转换过程会比常规的进制转换复杂一些,例如把1,0,0转换为10进制的过程是1*(2*3)+0*(2)+0=6;

          因此我们马上改变思路,采用类似大数算法的方法,即使用一个数组来表示数字,然后把每一位使用的进制存储到一个数组r中,即r中存储前25个素数。然后计算a+b的过程和大数加法完全一致,唯一不同的是,大数加法的每一位都使用10进制,而这里每一位使用的进制都不同。

          我们首先写出前25个素数,并存储到一个固定的数组中,我们可以用下面的代码来获取(请注意下面的代码完全是辅助性质,和最终题目的解并无关系):

img_1c53668bcee393edac0d7b3b3daff1ae.gif img_405b18b4b6584ae338e0f6ecaf736533.gif Code_求出前25个素数
#include <stdio.h>
#include 
<stdlib.h>
#include 
<math.h>

/* 判断一个数字n是否是素数 */
int isPrime(int n)
{
    
int i;
    
for( i=2; i < min( sqrt(n)+1, n-1) ; i++ )
    {
        
if(n%i==0)
            
return 0;
    }
    
return 1;
}

void main()
{
    
int n=1, count=0;
    printf(
"\n");
    
for( ; count <= 30; n++)
    {
        
if(isPrime(n))
        {
            printf(
"%3d,",n);
            count
++;
            
if(count%10==0)
                printf(
"\n");
        }
    }
}

          下面我们给出ZOJ 2529题目的完整代码:

img_1c53668bcee393edac0d7b3b3daff1ae.gif img_405b18b4b6584ae338e0f6ecaf736533.gif Code_ZOJ_2529_Solution
/* ZOJ 2529 - A+B in Hogwarts */
#include 
<stdio.h>
#include 
<stdlib.h>
#include 
<string.h>

/* 前25个质数,也是第i位的radix */
int r[]={2,3,5,7,1113,17,19,23,2931,37,41,43,4753,59,61,67,7173,79,83,89,97101,103,107,109,113};
/* 记a+b=c */
int a[30], b[30], c[30];
/*用于读入两个加数的缓冲区*/
char buf1[512], buf2[512];

/*根据输入的用逗号间隔的字符串,转化为魔法世界进制的数组表示*/
void ParseInput(char* buffer, int dest[])
{
    
int i=0;/*第i位*/
    
char *s;/*逗号所在的位置,从字符串的尾部向前查找*/
    
while( (s=strrchr(buffer,','))!=NULL )
    {
        
*s=0;/*在逗号处截断字符串!*/
        dest[i
++= atoi(s+1);
    }
    
/*已经没有逗号了,还剩下最高位的数字*/
    dest[i
++= atoi(buffer);
}

/*计算出c=a+b*/
void Add()
{
    
int i;
    
for(i=0;i<30;i++)
        c[i]
=a[i]+b[i];
    
/*进位*/
    
for(i=0;i<29;i++)
    {
        c[i
+1]+=c[i]/r[i];
        c[i]
=c[i]%r[i];
    }    
}

/*把结果打印出来,包含换行符,注意特殊情况,比如结果为1和0时*/
void PrintResult()
{
    
int i=29;
    
while(c[i]==0) i--;
    
for(;i>0;i--) printf("%d,", c[i]);
    
/*打印最后一位*/
    printf(
"%d\n", c[0]);
}

int main()
{
    
while(scanf("%s %s",buf1,buf2)!=EOF && strcmp(buf1,"end")!=0)
    {
        memset(a,
0,sizeof(a));
        memset(b,
0,sizeof(b));
        memset(c,
0,sizeof(c));
        ParseInput(buf1, a);
        ParseInput(buf2, b);
        Add();
        PrintResult();
    }
    
return 0;
}

 

          最后值得一提的是,解析输入的代码:ParseInput方法,用于把形如"10,6,4,2,0"这样的字符串解释到一个int数组(dest)内,是dest数组成为{0,2,4,6,10,0,...}。

          用于把结果输入的方法,PrintResult,要注意当计算结果为0和1这样的比较特殊的情况,必须也应该能正确输出。这些比较临界的细节是尤其需要注意的地方。

目录
相关文章
|
11月前
|
Java C++
poj 1503 高精度加法
把输入的数加起来,输入0表示结束。 先看我Java代码,用BigINteger类很多东西都不需要考虑,比如前导0什么的,很方便。不过java效率低点,平均用时600ms,C/C++可以0ms过。
33 1
|
11月前
|
图形学 C++
ZOJ1117 POJ1521 HDU1053 Huffman编码
Huffman编码的思想就是贪心,我们这里使用stl里的优先队列,priority_queue使用堆进行优化,虽然自己也可以写一个堆,但我感觉对于这道题有点主次不分了,再次感觉到stl确实是一个很强大的东西。
45 0
HDU-1370,Biorhythms(中国剩余定理)
本题主要就是应用中国剩余定理。
ZOJ 1403&&HDU 1015 Safecracker【暴力】
Safecracker Time Limit: 2 Seconds      Memory Limit: 65536 KB === Op tech briefing, 2002/11/02 06:42 CST === "The item is locked in a Klein safe behind a painting in the second-floor library.
1207 0