当前位置:   article > 正文

查缺补漏C语言:scanf( )的高阶用法_scanf在c语言中测长度函数

scanf在c语言中测长度函数

1) 指定读取长度

还记得在 printf() 中可以指定最小输出宽度吗?就是在格式控制符的中间加上一个数字,例如,%10d表示输出的整数至少占用 10 个字符的位置:

  • 如果整数的宽度不足 10,那么在左边以空格补齐;
  • 如果整数的宽度超过了 10,那么以整数本身的宽度来输出,10 不再起作用。
  1. #include <stdio.h>
  2. int main()
  3. {
  4. int a=10;
  5. printf("%10d",a);
  6. return 0;
  7. }

 

 scanf() 也有类似的用法,也可以在格式控制符的中间加一个数字,用来表示读取数据的最大长度,例如:
%2d表示最多读取两位整数;
%10s表示读取的字符串的最大长度为 10,或者说,最多读取 10 个字符。

  1. #include <stdio.h>
  2. int main()
  3. {
  4. int a;
  5. float b;
  6. char str[30];
  7. scanf("%2d",&a);
  8. scanf("%*[^\n]");scanf("%*c");
  9. scanf("%3f",&b);
  10. scanf("%*[^\n]");scanf("%*c");
  11. scanf("%5s",&str);
  12. printf("a=%d\nb=%f\nstr=%s\n",a,b,str);
  13. return 0;
  14. }

 

这段代码使用了多个 scanf() 函数连续读取数据,为了避免受到缓冲区中遗留数据的影响,每次读取结束我们都使用scanf("%*[^\n]"); scanf("%*c");来清空缓冲区。

限制读取数据的长度在实际开发中非常有用,最典型的一个例子就是读取字符串:我们为字符串分配的内存是有限的,用户输入的字符串过长就存放不了了,就会冲刷掉其它的数据,从而导致程序出错甚至崩溃;如果被黑客发现了这个漏洞,就可以构造栈溢出攻击,改变程序的执行流程,甚至执行自己的恶意代码,这对服务器来说简直是灭顶之灾。

在用 gets() 函数读取字符串的时候,有一些编译器会提示不安全,建议替换为 gets_s() 函数,就是因为 gets() 不能控制读取到的字符串的长度,风险极高。

就目前学到的知识而言,虽然 scanf() 可以控制字符串的长度,但是字符串中却不能包含空白符,这是硬伤,所以 scanf() 暂时还无法替代 gets()。不过大家也不要着急,稍后我还会补充 scanf() 的高级用法,届时 scanf() 就可以完全替代 gets(),并且比 gets() 更加智能。 

 

2) 匹配特定的字符

%s 控制符会匹配除空白符以外的所有字符,它有两个缺点:

  • %s 不能读取特定的字符,比如只想读取小写字母,或者十进制数字等,%s 就无能为力;
  • %s 读取到的字符串中不能包含空白符,有些情况会比较尴尬,例如,无法将多个单词存放到一个字符串中,因为单词之间就是以空格为分隔的,%s 遇到空格就读取结束了。


要想解决以上问题,可以使用 scanf() 的另外一种字符匹配方式,就是%[xxx][ ]包围起来的是需要读取的字符集合。例如,%[abcd]表示只读取字符abcd,遇到其它的字符就读取结束;注意,这里并不强调字符的顺序,只要字符在 abcd 范围内都可以匹配成功,所以你可以输入 abcd、dcba、ccdc、bdcca 等。

  1. #include <stdio.h>
  2. int main(){
  3. char str[30];
  4. scanf("%[abcd]", str);
  5. printf("%s\n", str);
  6. return 0;
  7. }

 

 

使用连接符

为了简化字符集合的写法,scanf() 支持使用连字符 -来表示一个范围内的字符,例如 %[a-z]、%[0-9] 等。

连字符左边的字符对应一个 ASCII 码,连字符右边的字符也对应一个 ASCII 码,位于这两个 ASCII 码范围以内的字符就是要读取的字符。注意,连字符左边的 ASCII 码要小于右边的,如果反过来,那么它的行为是未定义的。

常用的连字符举例:
  • %[a-z]表示读取 abc...xyz 范围内的字符,也即小写字母;
  • %[A-Z]表示读取 ABC...XYZ 范围内的字符,也即大写字母;
  • %[0-9]表示读取 012...789 范围内的字符,也即十进制数字。

你也可以将它们合并起来,例如:
  • %[a-zA-Z]表示读取大写字母和小写字母,也即所有英文字母;
  • %[a-z-A-Z0-9]表示读取所有的英文字母和十进制数字;
  • %[0-9a-f]表示读取十六进制数字。
  1. #include <stdio.h>
  2. int main(){
  3. char str[30];
  4. scanf("%[a-zA-Z]", str); //只读取字母
  5. printf("%s\n", str);
  6. return 0;
  7. }

 

不匹配某些字符

假如现在有一种需求,就是读取换行符以外的所有字符,或者读取 0~9 以外的所有字符,该怎么实现呢?总不能把剩下的字符都罗列出来吧,一是麻烦,二是不现实。

C语言的开发者们早就考虑到这个问题了,scanf() 允许我们在%[ ]中直接指定某些不能匹配的字符,具体方法就是在不匹配的字符前面加上^,例如:

  • %[^\n]表示匹配除换行符以外的所有字符,遇到换行符就停止读取;
  • %[^0-9]表示匹配除十进制数字以外的所有字符,遇到十进制数字就停止读取。

 

  1. #include <stdio.h>
  2. int main(){
  3. char str1[30], str2[30];
  4. scanf("%[^0-9]", str1);
  5. scanf("%*[^\n]"); scanf("%*c"); //清空缓冲区
  6. scanf("%[^\n]", str2);
  7. printf("str1=%s \nstr2=%s\n", str1, str2);
  8. return 0;
  9. }

 

 请注意第 6 行代码,它的作用是读取一行字符串,和 gets() 的功能一模一样。你看,scanf() 也能读取带空格的字符串呀,谁说 scanf() 不能完全取代 gets(),这明显是错误的说法。

另外,scanf() 还可以指定字符串的最大长度,指定字符串中不能包含哪些字符,这是 gets() 不具备的功能。

例如,读取一行不能包含十进制数字的字符串,并且长度不能超过 30:

  1. #include <stdio.h>
  2. int main(){
  3. char str[31];
  4. scanf("%30[^0-9\n]", str);
  5. printf("str=%s\n", str);
  6. return 0;
  7. }

 

总之,scanf() 不仅可以完全替代 gets(),并且比 gets() 的功能更加强大。 

3) 丢弃读取到的字符

在前面的代码中,每个格式控制符都要对应一个变量,把读取到的数据放入对应的变量中。其实你也可以不这样做,scanf() 允许把读取到的数据直接丢弃,不往变量中存放,具体方法就是在 % 后面加一个*,例如:

  • %*d表示读取一个整数并丢弃;
  • %*[a-z]表示读取小写字母并丢弃;
  • %*[^\n]表示将换行符以外的字符全部丢弃。
  1. #include <stdio.h>
  2. int main(){
  3. int n;
  4. char str[30];
  5. scanf("%*d %d", &n);
  6. scanf("%*[a-z]");
  7. scanf("%[^\n]", str);
  8. printf("n=%d, str=%s\n", n, str);
  9. return 0;
  10. }

 

 

 

 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/代码吟游诗人/article/detail/62628
推荐阅读
相关标签
  

闽ICP备14008679号