【C语言篇】scanf和printf万字超详细介绍(基本加拓展用法)(下篇)

简介: scanf处理⽤⼾输⼊的原理是,⽤⼾的输⼊先放⼊缓存,等到按下回⻋键后,按照占位符对缓存进⾏解读。 解读⽤⼾输⼊时,会从上⼀次解读遗留的第⼀个字符开始,直到读完缓存,或者遇到第⼀个不符合条件的字符为⽌。

【C语言篇】scanf和printf万字超详细介绍(基本加拓展用法)(上篇):https://developer.aliyun.com/article/1590496?spm=a2c6h.13148508.setting.17.1ee34f0eicv0Dx

scanf


当我们有了变量,我们需要给变量输⼊值就可以使⽤ scanf 函数,如果需要将变量的值输出在屏幕上的时候可以使⽤ prinf 函数,下⾯看⼀个例⼦:


#include <stdio.h>
int main()
{
 int score = 0;
 printf("请输⼊成绩:");
 scanf("%d", &score);
 printf("成绩是:%d\n", score);
 return 0;
}


注:标准输⼊⼀般指的就是键盘,标准输出⼀般指的就是屏幕


基本用法


scanf 函数⽤于读取⽤⼾的键盘输⼊。 程序运⾏到这个语句时,会停下来,等待⽤⼾从键盘输⼊。 ⽤⼾输⼊数据、按下回⻋键后, scanf就会处理⽤⼾的输⼊,将其存⼊变量。


它的原型定义在头⽂件 stdio.h 。 scanf的语法跟 printf类似。


int scanf ( const char * format, ... );
  • 同样也是可变参数类型
scanf("%d", &i);

它的第⼀个参数是⼀个格式字符串,⾥⾯会放置占位符(与 printf 的占位符基本⼀致),告诉编译器如何解读⽤⼾的输⼊,需要提取的数据是什么类型。


这是因为C语⾔的数据都是有类型的, scanf 必须提前知道⽤⼾输⼊的数据类型,才能处理数据。


它的其余参数就是存放⽤⼾输⼊的变量,格式字符串⾥⾯有多少个占位符,就有多少个变量。


上⾯⽰例中, scanf 的第⼀个参数 %d ,表⽰⽤⼾输⼊的应该是⼀个整数。 %d 就是⼀个占位 符, % 是占位符的标志, d 表⽰整数。第⼆个参数 &i 表⽰,将⽤⼾从键盘输⼊的整数存⼊变量 i


注意:变量前⾯必须加上 & 运算符即取地址运算符(指针变量除外),因为 scanf 传递的不是值,⽽是地址, 即将变量 i 的地址指向⽤⼾输⼊的值。 如果这⾥的变量是指针变量(⽐如字符串变量),那就不⽤加 & 运算符。


下⾯是⼀次将键盘输⼊读⼊多个变量的例⼦。


scanf("%d%d%f%f", &i, &j, &x, &y);

scanf处理数值占位符时,会⾃动过滤空⽩字符,包括空格、制表符、换⾏符等。


所以,⽤⼾输⼊的数据之间,有⼀个或多个空格不影响 scanf() 解读数据。另外,⽤⼾使⽤回⻋键,将输⼊分成⼏⾏,也不影响解读。

1
-20
3.4
-4.0e3

上⾯⽰例中,⽤⼾分成四⾏输⼊,得到的结果与⼀⾏输⼊是完全⼀样的。每次按下回⻋键以后, scanf 就会开始解读,如果第⼀⾏匹配第⼀个占位符,那么下次按下回⻋键时,就会从第⼆个占位符开始解读。


scanf处理⽤⼾输⼊的原理是,⽤⼾的输⼊先放⼊缓存,等到按下回⻋键后,按照占位符对缓存进⾏解读。 解读⽤⼾输⼊时,会从上⼀次解读遗留的第⼀个字符开始,直到读完缓存,或者遇到第⼀个不符合条件的字符为⽌。


#include <stdio.h>
int main()
{
    int x;
    float y;

    // ⽤⼾输⼊ " -13.45e12# 0" 
    scanf("%d", &x);
    printf("%d\n", x);
    scanf("%f", &y);
    printf("%f\n", y);
    return 0;
}

上⾯⽰例中, scanf 读取⽤⼾输⼊时, %d 占位符会忽略起⾸的空格,从 - 处开始获取数据,读取到 -13 停下来,因为后⾯的 . 不属于整数的有效字符。这就是说,占位符 %d 会读到 -13 ,存入x中。


第⼆次调⽤ scanf 时,就会从上⼀次停⽌解读的地⽅,继续往下读取。这⼀次读取的⾸字符 是 . ,由于对应的占位符是 %f ,会读取到 .45e12 ,这是采⽤科学计数法的浮点数格式,存入y中。


后⾯的 # 不属于浮点数的有效字符,所以会停在这里。


由于 scanf可以连续处理多个占位符,所以上⾯的例⼦也可以写成下⾯这样。


#include <stdio.h>
int main()
{
    int x;
    float y;
    // ⽤⼾输⼊ " -13.45e12# 0" 
    scanf("%d%f", &x, &y);
    return 0;
}

占位符


scanf常⽤的占位符如下,与 printf 的占位符基本⼀致。


  • %c :字符。
  • %d :整数。
  • %f : float 类型浮点数。
  • %lf : double 类型浮点数。
  • %Lf : long double 类型浮点数。
  • %s :字符串。
  • %[] :在⽅括号中指定⼀组匹配的字符(⽐如 %[0-9] ),遇到不在集合之中的字符,匹配将会停⽌。

**上⾯所有占位符之中,除了 %c 以外,都会⾃动忽略起⾸的空⽩字符。 %c 不忽略空⽩字符,总是返回当前第⼀个字符,⽆论该字符是否为空格。 **


如果要强制跳过字符前的空⽩字符,可以写成 scanf(" %c", &ch) ,即 %c 前加上⼀个空格,表⽰跳过零个或多个空⽩字符。


下⾯要特别说⼀下占位符 %s ,它不能简单地等同于字符串。


它的规则是,从当前第⼀个⾮空⽩字符开始读起,直到遇到空⽩字符(即空格、换⾏符、制表符等)为⽌。


因为 %s 不会包含空⽩字符,所以⽆法⽤来读取多个单词,除⾮多个 %s ⼀起使⽤。这也意味着, scanf不适合读取可能包含空格的字符串,⽐如书名或歌曲名。另外, scanf 遇到 %s 占位 符,会在字符串变量末尾存储⼀个空字符 '\0' 。


scanf不安全


**scanf 将字符串读⼊字符数组时,不会检测字符串是否超过了数组⻓度。**所以,储存字符串时, 很可能会超过数组的边界,导致预想不到的结果。为了防⽌这种情况,使⽤ %s 占位符时,应该指定读⼊字符串的最⻓⻓度,即写成 %[m]s ,其中的 [m] 是⼀个整数,表⽰读取字符串的最⼤⻓度,后⾯的字符将被丢弃


#include <stdio.h>
int main()
{
 char name[11];
 scanf("%10s", name);
 return 0;
}

上⾯⽰例中, name 是⼀个⻓度为11的字符数组, scanf 的占位符 %10s 表⽰最多读取⽤⼾输⼊的10个字符,后⾯的字符将被丢弃,这样就不会有数组溢出的⻛险了。


赋值忽略符

有时,⽤⼾的输⼊可能不符合预定的格式。

#include <stdio.h>
int main()
{
    int year = 0;
    int month = 0;
    int day = 0;
    scanf("%d-%d-%d", &year, &month, &day);
    printf("%d %d %d\n", year, month, day);
    return 0;
}

上⾯⽰例中,如果⽤⼾输⼊ 2020-01-01 ,就会正确解读出年、⽉、⽇。问题是⽤⼾可能输⼊其他格式,⽐如 2020/01/01 ,这种情况下, scanf 解析数据就会失败。


为了避免这种情况, scanf 提供了⼀个赋值忽略符(assignmentsuppression character) * 。 只要把 * 加在任何占位符的百分号后⾯,该占位符就不会返回值,解析后将被丢弃。


int main()
{
 int year = 0;
 int month = 0;
 int day = 0;
 scanf("%d%*c%d%*c%d", &year, &month, &day);
 return 0;
}

上面示例中, %*c 就是在占位符的百分号后⾯,加⼊了赋值忽略符 * ,表⽰这个占位符没有对应的变量,解读后不必返回。


返回值


scanf 的返回值是⼀个整数,表⽰成功读取的变量个数



如果没有读取任何项,或者匹配失败,则返回 0 。


如果在成功读取任何数据之前,发⽣了读取错误或者遇到读取到⽂件结尾,则返回常量EOF(-1)。


EOF-end of file⽂件结束标志

include <stdio.h>
    int main()
{
    int a = 0;
    int b = 0;
    float f = 0.0f;
    int r = scanf("%d %d %f", &a, &b, &f);
    printf("a=%d b=%d f=%f\n", a, b, f);
    printf("r = %d\n", r);
    return 0;
}

输⼊输出测试:


如果输⼊2个数后,按 ctrl+z ,提前结束输⼊:



在VS环境中按3次 ctrl+z ,才结束了输⼊,我们可以看到r是2,表⽰正确读取了2个数值。 如果⼀个数字都不输⼊,直接按3次 ctrl+z ,输出的r是-1,也就是EOF



scanf扩展用法

上面占位符讲到:


%[] :在⽅括号中指定⼀组匹配的字符(⽐如 %[0-9] ),遇到不在集合之中的字符,匹配将会停⽌。


匹配特定字符


  • 比如要匹配所有小写字母,可以写成%[a-z],匹配所有大字字符,可以写成%[A-Z](需要注意的是左边的字符要小于右边的字符),利用这种写法可以很方便的完成一些特定输入的读取
  • 遇到不符合的结束读取

读取大小写:


char s[20];
scanf("%[a-zA-Z]", s);
puts(s);

对于有空格的字符串如果想继续读,就需要添加一个空格(加在中间更清晰,但开头末尾也可以)

scanf("%[a-z A-Z]", s);
scanf("%[a-zA-Z ]", s);
scanf("%[ a-zA-Z]", s);


可以发现通过这种方法我们就可以通过scanf来读取连续字符串了


过滤字符


不匹配某些字符只需要在[ ]内的最前面加一个^就可以了,例如:

char ss[20];
scanf("%[^0-9]", s);
puts(s);

scanf("%[^0-9]", s)表示匹配除0-9之外的所有字符,遇到0-9的数字就结束读取

char s[20];
scanf("%[^\n]", s);
puts(s);

scanf("%[^\n]", s);表示匹配除\n的所有字符,遇到\n就结束读取


通过这种方法可以代替gets函数了


丢弃特定字符


就是上面讲到的赋值忽略符*这里结合[]使用

scanf("%*[a-z]");
scanf("%*[0-9]");

scanf("%*[a-z]")表示将读取到的小写字母丢弃

scanf("%*[0-9]")表示将读到的数字0-9丢弃


注意:

这种丢弃只发生在不满足丢弃条件的第一个字符之前

int main()
{
  char s[20];
  scanf("%*[a-z]%s",s);
  puts(s);
  return 0;
}


第一个不满足丢弃条件的是字符’A’,其之前的ab被丢弃了,但后面的ab还是会被读取

目录
相关文章
|
3月前
|
C语言
C语言判断逻辑的高阶用法
在C语言中,高级的判断逻辑技巧能显著提升代码的可读性、灵活性和效率。本文介绍了六种常见方法:1) 函数指针,如回调机制;2) 逻辑运算符组合,实现复杂条件判断;3) 宏定义简化逻辑;4) 结构体与联合体组织复杂数据;5) 递归与分治法处理树形结构;6) 状态机管理状态转换。通过这些方法,可以更高效地管理和实现复杂的逻辑判断,使代码更加清晰易懂。
231 88
|
2月前
|
C语言
初识C语言:与计算机的交流之输入与输出(scanf和printf)
初识C语言:与计算机的交流之输入与输出(scanf和printf)
182 0
|
3月前
|
存储 C语言 数据格式
【C语言基础考研向】03混合运算和printf讲解
本文分为两部分。第一部分介绍了C语言中的混合运算与类型强制转换的重要性,通过实例展示了当整型数进行除法运算且结果为小数时,必须使用类型转换才能正确存储浮点数结果。第二部分详细讲解了`printf`函数的功能与使用方法,包括格式化输出不同类型数据的基本语法,并通过具体示例演示了如何利用字段宽度和对齐方式来控制输出格式,帮助读者更好地理解和掌握输出格式的控制技巧。
49 10
|
3月前
|
C语言
【C语言基础考研向】05 scanf读取标准输入超详解
本文详细解析了C语言中`scanf`函数的工作原理及常见问题。首先介绍了`scanf`如何处理标准输入,并通过示例说明了为何有时会出现阻塞现象及其解决办法。接着探讨了当输入包含多种数据类型时,特别是字符型数据的处理方式,强调了格式控制的重要性,并给出了正确的输入格式示例。通过正确配置,可以避免因空格和换行符导致的问题,确保数据准确读取。
82 10
|
3月前
|
存储 C语言
【C语言基础考研向】10 字符数组初始化及传递和scanf 读取字符串
本文介绍了C语言中字符数组的初始化方法及其在函数间传递的注意事项。字符数组初始化有两种方式:逐个字符赋值或整体初始化字符串。实际工作中常用后者,如`char c[10]=&quot;hello&quot;`。示例代码展示了如何初始化及传递字符数组,并解释了为何未正确添加结束符`\0`会导致乱码。此外,还讨论了`scanf`函数读取字符串时忽略空格和回车的特点。
|
3月前
|
C语言
C语言程序设计核心详解 第三章:顺序结构,printf(),scanf()详解
本章介绍顺序结构的基本框架及C语言的标准输入输出。程序从`main()`开始依次执行,框架包括输入、计算和输出三部分。重点讲解了`printf()`与`scanf()`函数:`printf()`用于格式化输出,支持多种占位符;`scanf()`用于格式化输入,需注意普通字符与占位符的区别。此外还介绍了`putchar()`和`getchar()`函数,分别用于输出和接收单个字符。
|
2月前
|
C语言 C++
C语言 之 内存函数
C语言 之 内存函数
36 3
|
C语言
c语言中scanf()、printf()函数
  函数调用scanf(“%d”,  &weight) 包含两个参数:“%d” 和&weight。C用逗号来隔开函数调用中的多个参数; 但是printf()和scanf()函数比较特殊,其函数数目可以不受控制。
869 0
|
18天前
|
C语言
c语言调用的函数的声明
被调用的函数的声明: 一个函数调用另一个函数需具备的条件: 首先被调用的函数必须是已经存在的函数,即头文件中存在或已经定义过; 如果使用库函数,一般应该在本文件开头用#include命令将调用有关库函数时在所需要用到的信息“包含”到本文件中。.h文件是头文件所用的后缀。 如果使用用户自己定义的函数,而且该函数与使用它的函数在同一个文件中,一般还应该在主调函数中对被调用的函数做声明。 如果被调用的函数定义出现在主调函数之前可以不必声明。 如果已在所有函数定义之前,在函数的外部已做了函数声明,则在各个主调函数中不必多所调用的函数在做声明
31 6
|
1天前
|
存储 缓存 算法
【C语言】内存管理函数详细讲解
在C语言编程中,内存管理是至关重要的。动态内存分配函数允许程序在运行时请求和释放内存,这对于处理不确定大小的数据结构至关重要。以下是C语言内存管理函数的详细讲解,包括每个函数的功能、标准格式、示例代码、代码解释及其输出。
19 6