1. scanf
当我们有了变量,我们需要给变量输⼊值就可以使⽤scanf
函数,如果需要将变量的值输出在屏幕上的时候可以使用 printf
函数,下⾯看⼀个例⼦:
#include <stdio.h>
int main()
{
int score = 0;
printf("请输⼊成绩:"); //先打印一串文字
scanf("%d", &score); //用来读取自己输入的一个数字,
printf("成绩是:%d\n", score); //打印自己输入的数字
return 0;
}
那接下来我们介绍⼀下 scanf
函数。
1.1 基本用法
scanf()
函数⽤于读取⽤⼾的键盘输⼊。
程序运⾏到scanf()
语句时,会停下来,等待⽤⼾从键盘输⼊。
⽤⼾输⼊数据、按下回⻋键后,scanf()
就会处理⽤⼾的输⼊,将其存⼊变量。
它的原型定义也在头⽂件 stdio.h
。
scanf()
的语法跟printf()
类似。
scanf("%d", &i); //专门用来读取用户输入的数据的
它的第⼀个参数是⼀个格式字符串,⾥⾯会放置占位符(与 printf()
的占位符基本⼀致),告诉编
译器如何解读⽤⼾的输⼊,需要提取的数据是什么类型。
这是因为C语⾔的数据都是有类型的,scanf()
必须提前知道⽤⼾输⼊的数据类型,才能处理数
据。
它的其余参数就是存放⽤⼾输⼊的变量,格式字符串⾥⾯有多少个占位符,就有多少个变量。
上⾯示例中, scanf()
的第⼀个参数%d
,表⽰⽤⼾输⼊的应该是⼀个整数。 %d
就是⼀个占位符, %
是占位符的标志,d
表⽰整数。第⼆个参数&i
表⽰,将⽤⼾从键盘输⼊的整数存⼊变量i
。
注意:变量前⾯必须加上&
运算符(指针变量除外),因为scanf()
传递的不是值,⽽是地址,
即将变量 i
的地址指向⽤⼾输⼊的值。
如果这⾥的变量是指针变量(⽐如字符串变量),那就不⽤加&
运算符。
下⾯是⼀次将键盘输⼊读⼊多个变量的例⼦。
scanf("%d%d%f%f", &i, &j, &x, &y); //变量和占位符一一对应
上⾯示例中,格式字符串 %d%d%f%f
,表⽰⽤⼾输⼊的前两个是整数,后两个是浮点数,⽐如 1 -20 3.4 -4.0e3
(中间用空格隔开)。这四个值依次放⼊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" //整数读到小数点后就会停止,而小数读到特殊符号就会停止,e12表示10的二次方
scanf("%d", &x);
scanf("%f", &y);
return 0;
}
上⾯示例中, scanf()
读取⽤⼾输⼊时,%d
占位符会忽略起⾸的空格,从 - 处开始获取数据,读取到-13
停下来,因为后⾯的 . 不属于整数的有效字符。这就是说,占位符%d
会读到-13
。
第⼆次调⽤ scanf() 时,就会从上⼀次停⽌解读的地⽅,继续往下读取。这⼀次读取的⾸字符
是 .
,由于对应的占位符是%f
,会读取到.45e12
,这是采⽤科学计数法的浮点数格式。后面的#
不属于浮点数的有效字符,所以会停在这⾥。
由于 scanf()
可以连续处理多个占位符,所以上⾯的例⼦也可以写成下⾯这样。
#include <stdio.h>
int main()
{
int x;
float y;
// ⽤⼾输⼊ " -13.45e12# 0"
scanf("%d%f", &x, &y);
return 0;
}
1.2 scanf的返回值
scanf()
的返回值是⼀个整数,表示成功读取的变量个数。
如果没有读取任何项,或者匹配失败,则返回 0
。如果在成功读取任何数据之前,发生了读取错误或者遇到读取到⽂件结尾,则返回常量EOF。
#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
1.3 占位符
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()
将字符串读⼊字符数组时,不会检测字符串是否超过了数组⻓度。所以,储存字符串时,很可能会超过数组的边界,导致预想不到的结果。为了防⽌这种情况,使⽤%s
占位符时,应该指定读⼊字符串的最长长度,即写成 %[m]s
,其中的[m]
是⼀个整数,表⽰读取字符串的最⼤⻓度,后 ⾯的字符将被丢弃。
#include <stdio.h>
int main()
{
char name[11];
scanf("%10s", name);
return 0;
}
上⾯示例中, name
是⼀个⻓度为11的字符数组, scanf()
的占位符 %10s
表⽰最多读取⽤⼾输⼊ 的10个字符,后⾯的字符将被丢弃,这样就不会有数组溢出的风险了。
1.4 赋值忽略符 有时,⽤⼾的输⼊可能不符合预定的格式。
#include <stdio.h>
int main()
{
int year = 0;
int month = 0;
int day = 0;
scanf("%d-%d-%d", &year, &month, &day);
return 0;
}
上⾯示例中,如果⽤⼾输⼊ 2020-01-01
,就会正确解读出年、⽉、⽇。问题是⽤⼾可能输⼊其他格式,⽐如2020/01/01
,这种情况下,scanf()
解析数据就会失败。
为了避免这种情况, scanf()
提供了⼀个赋值忽略符(assignment suppression character) *
。
只要把 *
加在任何占位符的百分号后⾯,该占位符就不会返回值,解析后将被丢弃。
#include <stdio.h>
int main()
{
int year = 0;
int month = 0;
int day = 0;
scanf("%d%*c%d%*c%d", &year, &month, &day);
return 0;
}
上⾯示例中, %*c
就是在占位符的百分号后⾯,加⼊了赋值忽略符*
,表⽰这个占位符没有对应的 变量,解读后不必返回。