c/c++中字符输入问题

来源:互联网 发布:备孕前准备 知乎 编辑:程序博客网 时间:2024/06/06 16:47

在c和c++中,字符时我们用的非常多的一种数据类型,与使用其他基本数据类型不同的是,我们在进行字符操作的过程中,总会碰到这样那样的问题。
比如scanf会把回车当做一个字符读入。比如:
在用下面代码运行时:

#include<cstdio>#include<iostream>#include<algorithm>#include<cstring>#include<string>#include<cmath>#include<map>#include<set>#include<cctype>#include<ctime>#define INF 0x3f3f3f3f3#define PI acos(-1.0)using namespace std;char s[3][3];int main(){    int i,j;    for(i=0;i<3;i++)        for(j=0;j<3;j++)        scanf("%c",&s[i][j]);        for(i=0;i<3;i++)        {            for(j=0;j<3;j++)            cout<<s[i][j]<<' ';            cout<<endl;        }        return 0;}

结果为:

abcdefhija b c d ef hProcess returned 0 (0x0)   execution time : 11.271 sPress any key to continue.

这就是因为我们在每一次换行时输出的回车,是一个合法的字符,而且被数组接受了,导致正确的值没有被读入进去,我们在输出的时候选择输出字符的ascII值可以清楚的看出:

25     printf("%d ",s[i][j]);
abcdefghi97 98 9910 100 101102 10 103Process returned 0 (0x0)   execution time : 13.688 sPress any key to continue.

对于上述情况,我们有几种解决办法:
1.使用cin输入:

    for(i=0;i<3;i++)        for(j=0;j<3;j++)        //scanf("%c",&s[i][j]);        cin>>s[i][j];因为cin不会读入换行符,ps:博主听说scanfcin这种混合用法有时会发生玄学错误(缓冲区会出错),所以并不推荐,不过博主有时候为了方便也会这样用,orz。

2.使用getchar读取多余的换行符:

 for(i=0;i<3;i++)    {         for(j=0;j<3;j++)        scanf("%c",&s[i][j]);        getchar();    }

3.使用按行读入:

   for(i=0;i<3;i++)    {        scanf("%s",s[i]);//将二维字符数组看做一个一维数组,每一个元素为一个字符串    }

关于scanf和cin的讲解:
记得不久前看到一篇博客将这个讲的特别好的,然而忘记是哪里了,orz,刚刚找了一会没找到,啥时候找到了和大家分享。


突然学习了一个新知识:
在计算机还没有出现之前,有一种叫做电传打字机(Teletype Model 33)的玩意,每秒钟可以打10个字符。但是它有一个问题,就是打完一行换行的时候,要用去0.2秒,正好可以打两个字符。要是在这0.2秒里面,又有新的字符传过来,那么这个字符将丢失。

于是,研制人员想了个办法解决这个问题,就是在每行后面加两个表示结束的字符。一个叫做“回车”,告诉打字机把打印头定位在左边界;另一个叫做“换行”,告诉打字机把纸向下移一行。

这就是“换行”和“回车”的来历,从它们的英语名字上也可以看出一二。

后来,计算机发明了,这两个概念也就被般到了计算机上。那时,存储器很贵,一些科学家认为在每行结尾加两个字符太浪费了,加一个就可以。于是,就出现了分歧。

Unix系统里,每行结尾只有“<换行>”,即“\n”;Windows系统里面,每行结尾是“<换行><回车>”,即“\n\r”;Mac系统里,每行结尾是“<回车>”。一个直接后果是,Unix/Mac系统下的文件在Windows里打开的话,所有文字会变成一行;而Windows里的文件在Unix/Mac下打开的话,在每行的结尾可能会多出一个^M符号。

0 0
原创粉丝点击