C语言之如何用isspace()和ungetc()实现前导空白字符过滤

问题场景

当使用getchar()进行字符输入时,我们经常需要跳过输入流开头位置的空格、制表符、换行符等空白字符,直到遇到第一个有效字符为止。这在处理用户输入或解析文件时尤为常见。

对于初学者来说,在做算法练习的时候也会遇到如先为int类型的变量输入一个值,后再输入一个字符串的情况。此时由于不知道两次输入之间的空白字符的数量以及类型,处理前导的空字符便成了一个稍麻烦的问题。

洛谷许多题目进行不同行输入时通常采用的是”\r\n”而不是单纯的”\n”,楼主曾经就尝试使用游离的getchar()处理两次输入之间的’\n’,结果一道橙题折腾了一晚上也没AC…

关键函数说明

1. isspace()函数

#include <ctype.h>
int isspace(int c);
  • 判断传入的字符是否是空白字符
  • 支持的空白字符包括:空格(’ ‘)、换页符(’\f’)、换行符(‘\n’)、回车符(‘\r’)、水平制表符(‘\t’)、垂直制表符(‘\v’)

2. ungetc()函数

#include <stdio.h>
int ungetc(int c, FILE *stream);
  • 将指定字符推回输入流
  • 常用于”偷看”下一个字符后需要还原的场景
  • 最多保证1个字符的可靠回推

解决方案

#include <stdio.h>
#include <ctype.h>

void trim_leading_whitespace() {
    int c;
    // 跳过所有空白字符,注意这里while后面的分号
    while ((c = getchar()) != EOF && isspace(c));
    if (c != EOF) {
    	ungetc(c, stdin); // 将第一个非空白字符放回缓冲区
    }
}

使用示例:

int main() {
    printf("请输入带空格的字符串:");
    
    trim_leading_whitespace();
    
    int ch;
    printf("处理后的首字符:");
    if ((ch = getchar()) != EOF) {
        putchar(ch);
    }
    
    // 读取剩余字符
    printf("\n剩余字符:");
    while ((ch = getchar()) != '\n' && ch != EOF) {
        putchar(ch);
    }
    
    return 0;
}

运行效果:

请输入带空格的字符串:     Hello World
处理后的首字符:H
剩余字符:ello World

实现解析

  1. 循环读取:通过while循环持续读取字符,直到遇到EOF或非空白字符
  2. 字符判断:使用isspace()过滤所有类型的空白字符
  3. 字符回推:检测到第一个非空白字符后,使用ungetc()将其放回输入缓冲区
  4. 后续处理:主程序可以正常读取到过滤后的第一个有效字符

注意事项

  1. 回推限制:ANSI C保证至少1个字符的可靠回推,多个字符的回推行为取决于具体实现
  2. 流类型:适用于所有标准输入流(stdin)及文件流
  3. 错误处理:需考虑EOF的边界条件
  4. 编码兼容:完美处理ASCII编码,对宽字符需要改用iswspace()

通过这种组合使用,我们可以优雅地实现输入流的预处理,为后续的字符处理打下良好基础。

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

来源链接:https://www.jb51.net/program/339391rky.htm

© 版权声明
THE END
支持一下吧
点赞9 分享
评论 抢沙发
头像
请文明发言!
提交
头像

昵称

取消
昵称表情代码快捷回复

    暂无评论内容