解题思路:





注意事项:





参考代码:

几种解法不同的代码:

1.不需动脑的王牌代码,不需要字符串的对于初学者来说非常实用的一段代码。

#include<iostream>
    #include<cstdio>
    using namespace std;    int main(){        int a[5],first,second,third,fourth,fifth,sum=0;        cin>>a[1];
        getchar();        cin>>a[2];
        getchar();        cin>>a[3];
        getchar();        if(scanf("%d",&a[4])!=1)a[4]=10;
        sum+=a[1];
        first=a[2]/100;
        second=(a[2]/10)%10;
        third=a[2]%10;
        sum=sum+first*2+second*3+third*4;
        first=a[3]/10000;
        second=(a[3]/1000)%10;
        third=(a[3]/100)%10;
        fourth=(a[3]/10)%10;
        fifth=a[3]%10;
        sum=sum+first*5+second*6+third*7+fourth*8+fifth*9;
        sum=sum%11;        if(sum==a[4])cout<<"Right";        else{ if(sum!=10)printf("%d-%d-%d-%d",a[1],a[2],a[3],sum);                else printf("%d-%d-%d-X",a[1],a[2],a[3]);
        }        return 0;
    }

这个代码虽然有点繁琐,但确实能够AC。但是,一位大佬提出了一个问题:cin和getchar不能连用。本蒟蒻确实不知道。然后,作为一个字符串渣渣,我忽然想到了一种解决方式:在c++中,scanf有一个神奇的功能:可以跳过一些字符。 例如:scanf("%d,%d",&a,&b);,当输入1,2时,scanf跳过了“,”,把1赋给a,把2赋给b。现在,我们可以这样读入: scanf("%d-%d-%d-",&a[1],&a[2],&a[3]);这样,我们不利用getchar来读无用的“-”,而使用scanf的特性。这就是一个字符串蒟蒻的代码。
对于新手来说,还有一点小小的问题:if(scanf("%d",&a[4])!=1)a[4]=10;是什么意思?
实际上,当最后一位为x也就是字符时,把x赋给a[4]将不会成功。若赋值成功,scanf函数将会有一个返回值:1。所以,可以利用这个来判断是否是x,达到赋值的效果。
接下来便是一些取各位数字的方法,无需说明。

2.简单直白迅速的暴利模拟。

#include<iostream>
#include<cstdio>
using namespace std;
char a[14],b[14],t1,t2;int t22=0;
int main()
{
//    freopen("isbn.in","r",stdin);
//    freopen("isbn.ans","w",stdout);
    scanf("%c-%c%c%c-%c%c%c%c%c-%c",&a[1],&a[2],&a[3],&a[4],&a[5],&a[6],&a[7],&a[8],&a[9],&t1);//暴力读入//注意,利用了scanf的特性,可以读入指定的字符,具体可以自己查找资料。因此读入的都是有用字符
    for(int i=1;i<=9;++i)
        t22+=(a[i]-'0')*i;//暴力地安题目要求把有用的自负处理求和
    t2=t22%11+'0';//求出末位的标准值
    if(t2=='0'+10)t2='X';//特殊处理
    if(t1==t2)//判断,如果符合,输出Right
    {        cout<<"Right";        return 0;
    }    cout<<a[1]<<'-'<<a[2]<<a[3]<<a[4]<<'-'<<a[5]<<a[6]<<a[7]<<a[8]<<a[9]<<'-'<<t2;//暴力地输出
    return 0;//程序完美结束}

3.集合众多优点最为实用的代码之一,闪亮登场。

#include<iostream>
#include<string>
using namespace std;int main()
{    string isbncode;    
     int mathcode[13]={0},realcode=0;    
     long int mycode=0;    
     cin>>isbncode;//输入ISBN号 
    for(int i=0;i<=12;i++)//转换为数字 
        {
             mathcode[i]=isbncode[i]-'0'; //字符转换为数字 
        }
        mycode=mathcode[0]*1+mathcode[2]*2+mathcode[3]*3+mathcode[4]*4+mathcode[6]*5+mathcode[7]*6+mathcode[8]*7+mathcode[9]*8+mathcode[10]*9;
        mycode=mycode%11;//求出正确的识别码 
        if (mycode==10)
        {
            mycode='X'-'0';
        } 
        if(mycode==mathcode[12])//如果正确 
           cout<<"Right";//注意第一个字母大写,和题目要求一致 
        else
        {           //用一一对应的方式输出 
           cout<<mathcode[0]<<'-'
           <<mathcode[2]<<mathcode[3]<<mathcode[4]<<'-'
           <<mathcode[6]<<mathcode[7]<<mathcode[8]<<mathcode[9]<<mathcode[10]
           <<'-'<<char(mycode+'0');//用char函数强制转换为字符类型 
        }    return 0;
}

4.稍慢却好理解一些的一道直接模拟。

#include<iostream>
using namespace std;
int ans=0,k=1,check;char isbn[14]; //存ISBN码void get_in(){ //用来输入
    for(int i=1;i<=12;i++){    cin>>isbn[i];    if(isbn[i]!='-'){ //注意考虑-的情况
        ans+=(isbn[i]-'0')*k; //直接加起来
        k++; //用来累加的计数器
        }
    }    cin>>isbn[13];    if(isbn[13]!='X'){ //X单独处理
            check=isbn[13]-'0';
    }else check=10;
}void put(){ //输出函数
    if(ans%11==check){        cout<<"Right";
    }else{    
        for(int i=1;i<=12;i++) cout<<isbn[i];        if(ans%11!=10) cout<<ans%11; else cout<<"X"; //X单独处理
    }
}int main(){ //优雅的主程序
    get_in();
    put();    return 0;
}

5.较为常见的转换方法。

#include<iostream>
#include<cstring>
using namespace std;//头文件
int main(){    char s[20]; //isbn号码需要使用的字符串
    int i,ans=0,sum=0;    cin>>s;//输入isbn号码
    for(i=0;i<=11;i++){        if(s[i]>=48&&s[i]<=57){
            sum++;//数字的位数
            ans+=sum*(s[i]-48);//累乘计算mod前的值
        }        else continue;//遇到“-”不处理
    }
    ans%=11;//mod11得到isbn的识别码
    ans+=48;//转换为ASCII码
    if(ans==58)  ans='X';//处理“X”
    if(ans==s[12])  cout<<"Right"<<endl;//判断是否正确
    else {
      s[12]=ans;//将正确的值赋给最后一位
      for(i=0;i<=12;i++)  cout<<s[i];//输出正确的值
      cout<<endl;
    }    return 0;
}

6.和上面一种解法形成一个鲜明的对比,这是一种极其奇葩的解法。

#include<cstdio>
#include<cstring>//上面两个头文件不需要解释吧using namespace std;int main(){    char every[13];int sum=0,k=1;         //一个数组存数据,一个用来累加和,一个用来计算多少加了次(切记不能用i)
    for(int i=0;i<13;i++){
        every[i]=getchar();                       //读入数据
        if(every[i]!='-'&&i!=12) sum+=(every[i]-'0')*k++;         //进行判断:如果不是最后一个且不是分隔符'-'就累加
    }
    sum%=11;                            //所谓Mod 11的过程
    int tmp=every[12];                 //先保护之前的ISBN号
    every[12]=sum==10?'X':sum+'0';                     //写一个三目,判断是不是'X'也就是10
    if(tmp==every[12]) printf("Right");                   //如果不变,说明原来就是对的,输出"Right"
    else for(int i=0;i<13;i++){                                 //否则遍历一次数组全部输出来
        printf("%c",every[i]);
    }    return 0;                 //返回:0}

6.不服者来战,史上最简短的代码之一。

#include<stdio.h>
int main(void){  char a[14], mod[12] = "0123456789X"; //先将mod11后的十一个字符存入数组
  gets(a); //输入字符串
  int i, j = 1, t = 0;  for(i = 0; i < 12; i++) {        if(a[i] == '-') continue; //字符串为分隔符‘-’时跳过此次循环进入下一次循环
    t += (a[i]-'0')*j++; //t储存 第j个  数字  * j 的和
  }  if(mod[t%11] == a[12]) printf("Right");  else {
      a[12] = mod[t%11]; //若识别码错误,则赋正确的识别码,然后输出
      puts(a);
  }  return 0;
}

想啊想,怎么觉得的noi也就那样呢!也别怪我讽刺一下这个noi难度题!
对了,也别煞费了我的一番苦心,点个赞呗!!!!


点赞(9)
 

0.0分

2 人评分

C语言网提供由在职研发工程师或ACM蓝桥杯竞赛优秀选手录制的视频教程,并配有习题和答疑,点击了解:

一点编程也不会写的:零基础C语言学练课程

解决困扰你多年的C语言疑难杂症特性的C语言进阶课程

从零到写出一个爬虫的Python编程课程

只会语法写不出代码?手把手带你写100个编程真题的编程百练课程

信息学奥赛或C++选手的 必学C++课程

蓝桥杯ACM、信息学奥赛的必学课程:算法竞赛课入门课程

手把手讲解近五年真题的蓝桥杯辅导课程

评论列表 共有 1 条评论

pipi 5年前 回复TA
学到了,大佬喝可乐!