C语言空白字符读取问题

当我们读取字符或者字符串时,有时会遇到烦人的空白字符问题。因为结果很奇怪,为什么明明程序看起来没有错,但是结果读取到的值却是空的呢?下面我们来仔细分析一下。

问题一

在一个程序中连续读取多个变量时,如果最后一个是字符类型(char)的话,那么你一定要注意。

scanf()倾向于遗留下它“扫视”过但未读取的字符(包括换行符)。比如下面的程序:

    int i;
    int command;
    printf("Enter an interger: ");
    scanf("%d", &i);
    printf("Enter a command: ");

    command = getchar();
    // 或者是:
    //scanf("%c", &command);

在读入i的同时,scanf函数调用将会留下没有消耗掉的任意字符,包括(但不限于)换行符。getchar()或者scanf("%c", ...)将在随后取回第一个剩余字符,但这不是我们所希望的结果。

上述问题的解决方法之一就是可以在读取字符之前,先确保前面没有换行符。

    int i;
    int command;
    printf("Enter an interger: ");
    scanf("%d", &i);
    printf("Enter a command: ");

    while (getchar() != '\n')
        ;
    command = getchar();
    //scanf("%c", &command);

    printf("%d, %c\n", i, command);
}

上面的方法只是针对换行符,但是如果是其他空白字符的话,就有些麻烦了。在C标准库中有一个函数可以用来检测空白字符,我们需要包含头文件ctype.h

    int i;
    int command;
    printf("Enter an interger: ");
    scanf("%d", &i);
    printf("Enter a command: ");

    while (!isspace(getchar()))
        ;
    command = getchar();
    //scanf("%c", &command);

    printf("%d, %c\n", i, command);

问题二

假如我们要读取一行字符串,很多人都会采取类似于下面的方法:

#define MAX_LENGTH 100
int read_line(char str[])
{
    int ch, i = 0;
    while ((ch = getchar()) != '\n')
        if (i < MAX_LENGTH)
            str[i++] = ch;
    str[i] = '\0';
    return i;
}

NOTE:注意这里ch以及前面的command变量都为int类型,原因是getchar()函数实际上返回的是int类型,之所以这样设计是为了适应EOF这个特殊值。当这些函数读取出错或者读完文件后,会返回EOFEOF是一个宏,标准规定它的值必须是一个int型的负数常量。

上述方法看似没有任何问题,而且确实是没有问题。但是在某些情况下,上面的函数就会出现问题。假如在读入字符串之前,我先读入了某个变量并按下回车符或者其他空白字符,这时使用上面方法将什么也读不到或者读到的值开头将是空白字符。

原因很简单,read_line()函数中while循环结束的条件是读到换行符,如果前一次读取遗留下某些空白字符就会影响到后面的read_line()读取。

解决的方法很简单,下面是新的read_line()函数:

#define MAX_LENGTH 100
int read_line(char str[])
{
    int ch, i = 0;

    while (isspace(ch = getchar()))
        ;

    while ((ch = getchar()) != '\n' || ch != EOF)
        if (i < MAX_LENGTH)
            str[i++] = ch;
        ch = getchar();
    str[i] = '\0';
    return i;
}

我们同样使用了问题一中的isspace()函数。表达式isspace(ch = getchar())控制第一个字符,把读入的字符存储在ch中,然后使用isspace()函数判断ch是否是空白字符。如果不是,循环终止,ch中包含一个非空白字符。

参考资料

  1. C语言程序设计现代方法, K.N.King, 人民邮电出版社
  2. cplusplus.com:getchar

知识共享许可协议
本作品采用知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议进行许可。

  • 2
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值