题目链接
题目分析
(略)
解题思路
高精度运算 思路
AC程序(C++)
/**********************************
*@ID: 3stone
*@ACM: PAT. A1023 Have Fun with Numbers
*@Time: 17/2/28
*@IDE: DEV C++ 5.10
*@KEY:莫要好高骛远,你先成为软微最优秀的那部分再抱怨其他!!!
***********************************/
#include<cstdio>
#include<cstring>
using namespace std;
const int maxn = 1010;
//大整数类型 结构体
struct bign {
int d[maxn];
int len;
bign() { //构造函数初始化
len = 0;
memset(d, 0, sizeof(d));
}
};
//把字符串表示的大整数转换成bign类型
bign change(char str[]) {
bign c;
for(int i = strlen(str) - 1; i >= 0; i--) {
c.d[c.len++] = str[i] - '0';
}
return c;
}
//比较函数
int compare(bign a, bign b) {
if(a.len > b.len) return 1; //a大
else if(a.len < b.len) return -1; //a小
else {
for(int i = a.len - 1; i >= 0; i--)
if(a.d[i] > b.d[i]) return 1;
else if(a.d[i] < b.d[i]) return -1;
return 0; //相等
}
}
/*高精度加法
思路:每一步: 将该位上的两个数与进位相加,结果的个位作为该位结果,十位作为新的进位
预处理:若一正一负,就转为做高精度减法;若两个负数,则绝对值相加,结果再补负号;
*/
bign add(bign a, bign b) {
int carry = 0;
bign c;
for(int i = 0; i < a.len || i < b.len; i++) { //以较长的为界限(较短的高位全为0)
int temp = a.d[i] + b.d[i] + carry;
c.d[c.len++] = temp % 10;
carry = temp / 10;
}
if(carry != 0) { //最后还有进位,直接赋值给最高位
c.d[c.len++] = carry;
}
return c;
}
/* 高精度 * 低精度
思路:用高精度的每一位去乘低精度整体,再与进位相加,所得结果的个位作为该位结果,
其他高位所有部分作为新的进位
预处理:若有负数,则记录负号,取他们的绝对值带入运算
*/
bign multi(bign a, int b) {
int carry = 0;
bign c;
for(int i = 0; i < a.len; i++) {
int temp = b * a.d[i] + carry;
c.d[c.len++] = temp % 10;
carry = temp / 10;
}
while(carry != 0){ //乘法进位不止是个位数
c.d[c.len++] = carry % 10;
carry /= 10;
}
return c;
}
//输出
void print(bign a) {
for(int i = a.len - 1; i >= 0; i--)
printf("%d", a.d[i]);
}
int cnt[15];
int main() {
char str[25];
while(scanf("%s", str) != EOF) {
memset(cnt, 0, sizeof(cnt));
bign a = change(str);
for(int i = 0; i < a.len; i++) //记录每一个数出现的次数
cnt[a.d[i]]++;
a = multi(a, 2); //用加法也行!
for(int i = 0; i < a.len; i++){ //判断是否出现
if(cnt[a.d[i]] > 0) cnt[a.d[i]]--; //出现次数减 1
else{
printf("No\n");
print(a);
printf("\n");
return 0;
}
}
printf("Yes\n");
print(a);
printf("\n");
}
return 0;
}