『ACM C++』PTA浙大 | 基础题 - Have Fun with Numbers
连着这两道都是开学前数构老师的“爱心作业”,还没上课开学就给我们布置作业了,这道题有点小坑,也经常遇到类似的问题,特地拿出来记录一下。
------------------------------------------------题目----------------------------------------------------------
Notice that the number 123456789 is a 9-digit number consisting exactly the numbers from 1 to 9, with no duplication. Double it we will obtain 246913578, which happens to be another 9-digit number consisting exactly the numbers from 1 to 9, only in a different permutation. Check to see the result if we double it again!
Now you are suppose to check if there are more numbers with this property. That is, double a given number with k digits, you are to tell if the resulting number consists of only a permutation of the digits in the original number.
Input Specification:
Each input contains one test case. Each case contains one positive integer with no more than 20 digits.
Output Specification:
For each test case, first print in a line "Yes" if doubling the input number gives a number that consists of only a permutation of the digits in the original number, or "No" if not. Then in the next line, print the doubled number.
Sample Input:
Sample Output:
Yes
------------------------------------------------题目----------------------------------------------------------
(一) 原题大意:
123456789是一个9位数的数字,它完全由1到9之间的数字组成,没有重复。加倍,我们将得到246913578,这恰好是另一个9位数字,由1到9的数字组成,也是没有重复,只是排列不同。
简单来说,就是判断输入的数乘以2之后的结果,且结果的数和输入的书组成数字一致,且出现次数一致,例如1234567899,出现了两次9,翻倍之后2469135798,也出现两次9,则说明该数Yes,符合要求。
(二) 题目分析:
想法一:使用long long int存储输入的数值,然后翻倍,再用分割位数的方法对比输入数和结果数是否一致来判断。
注:这个想法是错误的,也就是我刚刚一开始说的坑点,下面贴上一个表
| 类型名称 | 字节数 | 取值范围 |
| signed char | 1 | -128~+127 (3位) |
| short int | 2 | -32768~+32767(5位) |
| int | 4 | -2147483648~+2147483647(10位) |
| long int | 4 | -2147483648~+2141483647(10位) |
| long long int | 8 | -9223372036854775808~+9223372036854775807(19位) |
我们可以清楚的发现long long int 最多只能达到19位,而题目输入要求写了 no more than 20 digits. 不会超过20位,也就是说long long int 存在超范围风险,是不成立的,而且翻倍乘以2之后是存在进位风险的呢。
想法二:既然用整型long long int都不够存的话,这时候就要使用字符串输入处理法了,将输入的每一位数都当成一个字符来处理,通过翻倍和进位来记录。这种方法的难度是在进位处理上,一不小心还是很容易错的。
(三) 代码分块:
想法一的错误代码我先贴上把,这种做法很简单,应该很多人都能想到:
#include<iostream>
#include<cmath>
#include<stdio.h>
using namespace std;
long long int Number;
long long int num,num2,sum = ,sum2 = ,k;
int Oct[];
int Oct2[];
int OK = ;
int main()
{
scanf("%lld",&Number);
num = Number;
num2 = *Number;
while(num>)
{
k = num % ;
Oct[k]++;
sum += k;
num = num / ;
}
while(num2>)
{
k = num2 % ;
Oct2[k]++;
sum2 += k;
num2 = num2 / ;
}
for(int i = ;i<;i++) if(Oct[i] != Oct2[i]) OK = ;
if(OK == )
{
printf("Yes\n");
printf("%lld",*Number);
}
else printf("No\n");
return ;
}
做法就是一个变量存输入数,另一个变量存翻倍数,然后分别分隔,然后再互相对比个数组成是否相同。
下面讲讲想法二AC的代码分块:
首先,先输入数值并获取它的长度:
scanf("%s", num1);
for (i = ; num1[i] == ; i--);
然后再通过对每一位进行翻倍进位:
for ( ; i >= ; i -- )
{
ji = (num1[i] - '') * ; //ji是翻倍后的结果
ans[num1[i] - ''] ++; //ans对原数相应位的个数++
di = ji % ; //*2后的当前位的数字
num2[i] = di + jin + '';
ans[num2[i] - ''] --; //ans对结果数的相应位的个数--
jin = (ji + jin) / ;
}
if (jin != 0) ans[jin] ++;
上面可以形成一个经典的大数计算模板,通过这道题我也开始接触到了一个新的问题的研究中 —— 大数的加减乘除的方法,过几天研究好了贴博客。
第三步就是判断ans是否全部都为0,若是,则说明原数和结果数是相同的排列:
for (i = ; i < ; i++)if (ans[i] != )break;//判断ans是否全部都为0,若是,则说明原数和结果数是相同的排列
if (i == ) printf("Yes");
else puts("No");
if (jin != )printf("%d", jin);
put(num2);
注:这里直接省略了for去输出num2结果数组,而是直接使用puts函数了,这个函数可以直接把数组所有内容直接输出。
(四) AC代码:
#include<stdio.h>
using namespace std;
int ans[];
char num1[];
char num2[];
int i,di = , jin = ,ji = ;
int main()
{
scanf("%s", num1);
for (i = ; num1[i] == ; i--);
for ( ; i >= ; i -- )
{
ji = (num1[i] - '') * ; //ji是翻倍后的结果
ans[num1[i] - ''] ++;//ans对原数相应位的个数++
di = ji % ;//*2后的当前位的数字
num2[i] = di + jin + '';
ans[num2[i] - ''] --;//ans对结果数的相应位的个数--
jin = (ji + jin) / ;
}
if (jin != ) ans[jin] ++;
for (i = ; i < ; i++)if (ans[i] != )break;//判断ans是否全部都为0,若是,则说明原数和结果数是相同的排列
if (i == ) printf("Yes");
else printf("No");
if (jin != )printf("%d", jin);
puts(num2);
return ;
}
(五)AC截图:

(六) 解后分析:
同样是一道练手的基础题,唯一坑点就是对long long int位数认知的模糊,导致一上手就没脑子的long long int,难点在于用char类型去处理数字基础运算,这也为我打开了一个新世界的大门,那就是大数的计算方法,待我整理整理再贴上来。
注:如果有更好的解法,真心希望您能够评论留言贴上您的代码呢~互相帮助互相鼓励才能成长鸭~~
『ACM C++』PTA浙大 | 基础题 - Have Fun with Numbers的更多相关文章
- 『ACM C++』PTA浙大 | 基础题 - 打印沙漏
<数据结构>开课前的一些小作业练习,可能因为一个寒假都没有打C++手生了,整个寒假都在帮拍电影做后期特效,导致这道题居然用了两个钟去AC,深感惭愧,作个标记吧,下面上题. 一首好曲推荐:同 ...
- 『ACM C++』 PTA 天梯赛练习集L1 | 021-024
忙疯警告,这两天可能进度很慢,下午打了一下午训练赛,训练赛的题我就不拿过来的,pta就做了一点点,明天又是满课的一天,所以进度很慢啦~ -------------------------------- ...
- 『ACM C++』 PTA 天梯赛练习集L1 | 007-011
真的是忙头晕了,学业.ACM打题.班级活动.自学新东西,哇这充实的大学~ ------------------------------------------------L1-007--------- ...
- 『ACM C++』 PTA 天梯赛练习集L1 | 036-037
这几天比较忙,所以随便做做水题了,得赶紧把英剧搞完啊啊啊啊啊啊 ------------------------------------------------L1-036-------------- ...
- 『ACM C++』 PTA 天梯赛练习集L1 | 001-006
应师兄要求,在打三月底天梯赛之前要把PTA上面的练习集刷完,所以后面的时间就献给PTA啦~ 后面每天刷的题都会把答案代码贡献出来,如果有好的思路想法也会分享一下~ 欢迎大佬提供更好的高效率算法鸭~ - ...
- 『ACM C++』 PTA 天梯赛练习集L1 | 029-033
哈哈,今天开始我也是学车人了~ 开始一千多道疯狂刷题~ ------------------------------------------------L1-029------------------ ...
- 『ACM C++』 PTA 天梯赛练习集L1 | 025-026
满课一天,做25的时候还疯狂WA,进度可以说是很慢了 哭泣 ------------------------------------------------L1-025---------------- ...
- 『ACM C++』 PTA 天梯赛练习集L1 | 018-020
终于一周有这么一天能够安静下来好好学习打打题,还是很美滋滋的哈哈~加油加油~ ------------------------------------------------L1-018------- ...
- 『ACM C++』 PTA 天梯赛练习集L1 | 052-053
今日刷题,水题水题 ------------------------------------------------L1-052------------------------------------ ...
随机推荐
- 【转】:Oracle Linux6.9下安装Oracle 11.2.0.4.0及psu补丁升级
为方便截图,本文操作都在vmware虚拟机上完成. 目录: 1.操作系统安装 2.数据库安装 3.PSU补丁升级卸载 part1 操作系统安装 Oracle (Enterprise) Linux ...
- 一个百度MAP导航的基础封装
项目中需要根据点击时候点击的内容,输入百度地图查找并展示规划等相关功能 于是封装了一个单独的百度map的html页面以供调用 功能包括了 ①展示底图 ②切换卫星图,切换卫星路线图,切换普通地图 ③通过 ...
- vue父子组件之间的传值
引入组件 父组件 <div> <form-edit></form-edit> </div> import FormEdit from "路径& ...
- 19-3-14Python中函数的进阶
1.动态参数: def func(*args): #在形参位置*叫做聚合 print(args) #元组形式 func(1,2,3,4,5,6) def func(**kwargs): # 动态关键字 ...
- mysql 常用的时间日期函数小结
本文主要是总结一些常用的在实际运用中常用的一些mysql时间日期以及转换的函数 1.now() :返回当前日期和时间 select now(); //2018-04-21 09:19:21 2.cu ...
- iview中tree的事件运用
iview中的事件和方法如下: 案例说明: html代码 <Tree :data="data4" @on-check-change="choiceAll" ...
- 安装Maven后使用cmd 执行 mvn -version命令 报错JAVA_HOME should point to a JDK not a JRE
1. 可以执行maven指令,说明maven的配置没错 2. 打开cmd,在cmd输入: set JAVA_HOME=D:\Program Files\Java\jdk1.8.0_91 3. 再测试是 ...
- 使用for in 循环数据集
在DELPHI没有FOR IN的语法时,我们要使用如下代码枚举数据集中的每个内容: cds.First; while not cds.eof do begin ... cds.Next; end; 最 ...
- ETL项目1:大数据采集,清洗,处理:使用MapReduce进行离线数据分析完整项目
ETL项目1:大数据采集,清洗,处理:使用MapReduce进行离线数据分析完整项目 思路分析: 1.1 log日志生成 用curl模拟请求,nginx反向代理80端口来生成日志. #! /bin/b ...
- 怎样在windows上定时执行python脚本
作为一个需要在电脑上工作和学习的人,一件十分困扰我的事情就是怎样不受互联网中其他内容的干扰而专注于自己想要做的事情,有的时候真的是沉浸于微博上的消息,忘了自己本来想要做的事.不过我有一件神器,自己爱豆 ...