科学计数法是科学家用来表示很大或很小的数字的一种方便的方法,其满足正则表达式 [±][1-9].[0-9]+E[±][0-9]+,即数字的整数部分只有 1 位,小数部分至少有 1 位,该数字及其指数部分的正负号即使对正数也必定明确给出。
现以科学计数法的格式给出实数 A,请编写程序按普通数字表示法输出 A,并保证所有有效位都被保留。
输入格式:
每个输入包含 1 个测试用例,即一个以科学计数法表示的实数 A。该数字的存储长度不超过 9999 字节,且其指数的绝对值不超过 9999。
输出格式:
对每个测试用例,在一行中按普通数字表示法输出 A,并保证所有有效位都被保留,包括末尾的 0。
输入样例 1:
+1.23400E-03
输出样例 1:
0.00123400
输入样例 2:
-1.2E+10
输出样例 2:
-12000000000
| 代码长度限制 | 时间限制 ||内存限制 |
|16 KB | 200 ms | 64 MB |
思路:
①有些题目的输入比较复杂,用常用的方式可能会难以处理,大家可以去学习一下关于正则表达式的基本知识,以及如何用scanf()去进行正则匹配,我看的是这个------>B站正则入门教程
②在scanf()中%[ ]
, 表示一个字符集, 如%[0-9]
表示只允许读入’0’到’9’之间的字符, %[a-zA-Z]
表示只读入字母, '-'
是范围连接符,也可以直接列出你需要读入的字符, 比如%[abcdef]
, 表示只允许读入在这6个字母当中的字符
③在解决完输入问题后只需要考虑小数点的移动问题就OK了, 就全是输出方面的考虑了
④输出先判断正负,负数要先输出负号
⑤根据指数的正负来判断如何输出,为负很简单,为正的话分为指数值>=小数部分长度
和指数值<小数部分长度
来处理,在代码的注释中给出了非常详细的解释
⑥一开始可能会觉得很难理解,建议在草稿纸上自己画一画去模拟小数点移动的过程,对照着思考非常有益于理解代码
代码:
#include<bits/stdtr1c++.h>
int main() {
int index;
char num[10005], chr;
scanf("%[+-]%[1-9].%[0-9]E%d", &chr, num, num + 1, &index); //用正则表达式控制输入,num表示从第0位开始存,num+1表示从第一位开始存
if (chr == '-') printf("-"); //负数的话上来先输出一个负号,正数不用管,下面分为指数<0和指数>0讨论
if (index < 0) {
for (int i = 0; i > index; i--) {
if (i == 0) printf("0.");
else printf("0");
} //如果指数为负,则相对简单,把前面的0输出后,直接输出整个字符数组即可,当然第一个0后面别忘记加小数点
printf("%s", num);
} else { //下面处理指数为正的情况
printf ("%c%c", num[0], num[1]); //先输出前两位
int j = 2; //下面需要从第三位开始输出
for (int i = 0; i < index - 1; i++) {
if (num[j] != '\0') printf("%c", num[j++]); //没到末尾就一直输出,char[]类型的字符数组末尾是'\0',但是不计入串长
else printf("0"); //到了末尾但是指数值大于原来小数部分长度的话,就在后面补0,比如+1.2034E+10,小数部分有四位,指数为10,,所以需要补6个0
}
if (j < int(strlen(num))) printf("."); //如果指数值小于原小数部分长度,则需要输出小数点,再用for循环输出余下的部分
for (; j < int(strlen(num)); j++) printf("%c", num[j]); //如+1.23456E+3,指数小于小数部分长度,通过上面的for循环输出1234后,在输出小数点后,再通过此for循环输出56
}
return 0;
}