洛谷 P1055 ISBN号码 字符串处理

洛谷 P1055 ISBN号码

题目描述
每一本正式出版的图书都有一个ISBN号码与之对应,ISBN码包括99位数字、11位识别码和33位分隔符,其规定格式如x-xxx-xxxxx-x,其中符号-就是分隔符(键盘上的减号),最后一位是识别码,例如0-670-82162-4就是一个标准的ISBN码。ISBN码的首位数字表示书籍的出版语言,例如00代表英语;第一个分隔符-之后的三位数字代表出版社,例如670670代表维京出版社;第二个分隔符后的五位数字代表该书在该出版社的编号;最后一位为识别码。

识别码的计算方法如下:

首位数字乘以11加上次位数字乘以22……以此类推,用所得的结果 \bmod 11mod11,所得的余数即为识别码,如果余数为1010,则识别码为大写字母XX。例如ISBN号码0-670-82162-4中的识别码44是这样得到的:对067082162这99个数字,从左至右,分别乘以1,2,…,91,2,…,9再求和,即0×1+6×2+……+2×9=1580×1+6×2+……+2×9=158,然后取158 \bmod 11158mod11的结果44作为识别码。

你的任务是编写程序判断输入的ISBN号码中识别码是否正确,如果正确,则仅输出Right;如果错误,则输出你认为是正确的ISBN号码。

输入格式
一个字符序列,表示一本书的ISBN号码(保证输入符合ISBN号码的格式要求)。

输出格式
一行,假如输入的ISBN号码的识别码正确,那么输出Right,否则,按照规定的格式,输出正确的ISBN号码(包括分隔符-)。

输入输出样例

输入 #1
0-670-82162-4
输出 #1
Right

输入 #2
0-670-82162-0
输出 #2
0-670-82162-4


分析

一道较为简单的字符串处理题, 按要求模拟即可, 但有几个收获点, 值得记下来, 先贴代码再具体分析


代码

#include
int main(void)
{
	char a[20];
	scanf("%s", a);
	int b[10];
	int j = 0, i = 0;
	for( i=0; a[i+1]!='\0'; i++)
	{
		if(a[i]>='0' && a[i]<='9')
		{
//			b[j++] = (int)a[i];		//我们看到 a[i]的位置存了一个'0', 其实存的是它的ASCII码——48
//			b[j++] = a[i]; 			//结果同上, 或许因为二者均存的是数字, 所以可以直接用char给int赋值
			b[j++] = a[i] - '0'; 
		}
	}
	int res;
	if( a[i]=='X')
	{
		res = 10;
	}
	else
	{
		res = a[i] - '0';
	}
	
	int ans = 0;
	for( j=0; j<9; j++)
	{
		ans += b[j]*(j+1);
	}
	ans %= 11;
	if( ans==res)
	{
		printf("Right");
	}
	else
	{
		if( ans == 10)
		{
			a[i] = 'X';
		}
		else
		{
			a[i] = ans + '0';
		}
		
		for( i=0; a[i]!='\0'; i++)
		{
			putchar(a[i]);
		}
	}
	return 0;
}

收获与反思

  1. 其实最重要的代码里已经注释过了, 大概就是把一个字符赋值给一个整型变量时, 会把该字符的ASCII码丢给这个变量. 举个例子就是char a[]中读入了01234, 再把他们按位丢给int b[]后, b中接收的值就是48, 49, 50, 51, 52, 而不需要所谓强制类型转换, 只需要把丢过来的数-48或-‘0’ 即可.
  2. 认真审题, 本题因为丢了"X"的情况而WA了两次. 尽量将特殊的地方进行标记, 或以注释的形式放在题头上, 保证自己不会忘
  3. 据说puts()不安全, 本题第一次尝试用putchar()输出结果, 算是明白怎么用了

你可能感兴趣的:(练习题,c++,字符串)