C语言scanf:读取从键盘输入的数据(含输入格式汇总表)
程序是人机交互的媒介,有输出必然也有输入,第三章我们讲解了如何将数据输出到显示器上,本章我们开始讲解如何从键盘输入数据。在C语言中,有多个函数可以从键盘获得用户输入:
- scanf():和 printf() 类似,scanf() 可以输入多种类型的数据。
- getchar()、getche()、getch():这三个函数都用于输入单个字符。
- gets():获取一行数据,并作为字符串处理。
scanf() 是最灵活、最复杂、最常用的输入函数,但它不能完全取代其他函数,大家都要有所了解。
本节我们只讲解 scanf(),其它的输入函数将在下节讲解。
scanf()函数
scanf 是 scan format 的缩写,意思是格式化扫描,也就是从键盘获得用户输入,和 printf 的功能正好相反。我们先来看一个例子:
运行结果:#include <stdio.h>
int main()
{
int a = 0, b = 0, c = 0, d = 0;
scanf("%d", &a); //输入整数并赋值给变量a
scanf("%d", &b); //输入整数并赋值给变量b
printf("a+b=%d\n", a+b); //计算a+b的值并输出
scanf("%d %d", &c, &d); //输入两个整数并分别赋值给c、d
printf("c*d=%d\n", c*d); //计算c*d的值并输出
return 0;
}
12↙
60↙
a+b=72
10 23↙
c*d=230
↙表示按下回车键。从键盘输入12,按下回车键,scanf() 就会读取输入数据并赋值给变量 a;本次输入结束,接着执行下一个 scanf() 函数,再从键盘输入 60,按下回车键,就会将 60 赋值给变量 b,都是同样的道理。第 8 行代码中,scanf() 有两个以空格分隔的
%d,后面还跟着两个变量,这要求我们一次性输入两个整数,并分别赋值给 c 和 d。注意"%d %d"之间是有空格的,所以输入数据时也要有空格。对于 scanf(),输入数据的格式要和控制字符串的格式保持一致。其实 scanf 和 printf 非常相似,只是功能相反罢了:
它们都有格式控制字符串,都有变量列表。不同的是,scanf 的变量前要带一个scanf("%d %d", &a, &b); // 获取用户输入的两个整数,分别赋值给变量 a 和 b
printf("%d %d", a, b); // 将变量 a 和 b 的值在显示器上输出
&符号。&称为取地址符,也就是获取变量在内存中的地址。在《数据在内存中的存储》一节中讲到,数据是以二进制的形式保存在内存中的,字节(Byte)是最小的可操作单位。为了便于管理,我们给每个字节分配了一个编号,使用该字节时,只要知道编号就可以,就像每个学生都有学号,老师会随机抽取学号来让学生回答问题。字节的编号是有顺序的,从 0 开始,接下来是 1、2、3……
下图是 4G 内存中每个字节的编号(以十六进制表示):

这个编号,就叫做地址(Address)。
int a;会在内存中分配四个字节的空间,我们将个字节的地址称为变量 a 的地址,也就是&a的值。对于前面讲到的整数、浮点数、字符,都要使用 & 获取它们的地址,scanf 会根据地址把读取到的数据写入内存。我们不妨将变量的地址输出看一下:
输出结果:#include <stdio.h>
int main()
{
int a='F';
int b=12;
int c=452;
printf("&a=%p, &b=%p, &c=%p\n", &a, &b, &c);
return 0;
}
&a=0x18ff48, &b=0x18ff44, &c=0x18ff40
%p是一个新的格式控制符,它表示以十六进制的形式(带小写的前缀)输出数据的地址。如果写作%P,那么十六进制的前缀也将变成大写形式。
图:a、b、c 的内存地址
注意:这里看到的地址都是假的,是虚拟地址,并不等于数据在物理内存中的地址。虚拟地址是现代计算机因内存管理的需要才提出的概念,我们将在《C语言内存精讲》专题中详细讲解。
再来看一个 scanf 的例子:
#include <stdio.h>
int main()
{
int a, b, c;
scanf("%d %d", &a, &b);
printf("a+b=%d\n", a+b);
scanf("%d %d", &a, &b);
printf("a+b=%d\n", a+b);
scanf("%d, %d, %d", &a, &b, &c);
printf("a+b+c=%d\n", a+b+c);
scanf("%d is bigger than %d", &a, &b);
printf("a-b=%d\n", a-b);
return 0;
}
运行结果:
10 20↙
a+b=30
100 200↙
a+b=300
56,45,78↙
a+b+c=179
25 is bigger than 11↙
a-b=14
个 scanf() 的格式控制字符串为
"%d %d",中间有一个空格,而我们却输入了10 20,中间有多个空格。第二个 scanf() 的格式控制字符串为"%d %d",中间有多个空格,而我们却输入了100 200,中间只有一个空格。这说明 scanf() 对输入数据之间的空格的处理比较宽松,并不要求空格数严格对应,多几个少几个无所谓,只要有空格就行。第三个 scanf() 的控制字符串为
"%d, %d, %d",中间以逗号分隔,所以输入的整数也要以逗号分隔。第四个 scanf() 要求整数之间以
is bigger than分隔。用户每次按下回车键,程序就会认为完成了一次输入操作,scanf() 开始读取用户输入的内容,并根据格式控制字符串从中提取有效数据,只要用户输入的内容和格式控制字符串匹配,就能够正确提取。
本质上讲,用户输入的内容都是字符串,scanf() 完成的是从字符串中提取有效数据的过程。
连续输入
在本节段示例代码中,我们一个一个地输入变量 a、b、c、d 的值,每输入一个值就按一次回车键。现在我们改变输入方式,将四个变量的值一次性输入,如下所示:
12 60 10 23↙
a+b=72
c*d=230
可以发现,两个 scanf() 都能正确读取。情理的猜测是,个 scanf() 读取完毕后没有抛弃多余的值,而是将它们保存在了某个地方,下次接着使用。如果我们多输入一个整数,会怎样呢?
12 60 10 23 99↙
a+b=72
c*d=230
这次我们多输入了一个 99,发现 scanf() 仍然能够正确读取,只是 99 没用罢了。如果我们少输入一个整数,又会怎样呢?
12 60 10↙
a+b=72
23↙
c*d=230
输入三个整数后,前两个 scanf() 把前两个整数给读取了,剩下一个整数 10,而第三个 scanf() 要求输入两个整数,一个单独的 10 并不能满足要求,所以我们还得继续输入,凑够两个整数以后,第三个 scanf() 才能读取完毕。从本质上讲,我们从键盘输入的数据并没有直接交给 scanf(),而是放入了缓冲区中,直到我们按下回车键,scanf() 才到缓冲区中读取数据。如果缓冲区中的数据符 scanf() 的要求,那么就读取结束;如果不符要求,那么就继续等待用户输入,或者干脆读取失败。我们将在本章的《进入缓冲区(缓存)的世界,破解一切与输入输出有关的疑难杂症》《结C语言缓冲区谈scanf函数》两节中详细讲解缓冲区。
注意,如果缓冲区中的数据不符 scanf() 的要求,要么继续等待用户输入,要么就干脆读取失败,上面我们演示了“继续等待用户输入”的情形,下面我们对代码稍作修改,演示“读取失败”的情形。
#include <stdio.h>
int main()
{
int a = 1, b = 2, c = 3, d = 4; //修改处:给变量赋予不同的初始值
scanf("%d", &a);
scanf("%d", &b);
printf("a=%d, b=%d\n", a, b);
scanf("%d %d", &c, &d);
printf("c=%d, d=%d\n", c, d);
return 0;
}
运行结果:
12 60 a10↙
a=12, b=60
c=3, d=4
前两个整数被正确读取后,剩下了 a10,而第三个 scanf() 要求输入两个十进制的整数,a10 无论如何也不符要求,所以只能读取失败。输出结果也证明了这一点,c 和 d 的值并没有被改变。这说明 scanf() 不会跳过不符要求的数据,遇到不符要求的数据会读取失败,而不是再继续等待用户输入。
总而言之,正是由于缓冲区的存在,才使得我们能够多输入一些数据,或者一次性输入所有数据,这可以认为是缓冲区的一点优势。然而,缓冲区也带来了一定的负面影响,甚会导致很奇怪的行为,请看下面的代码:
#include <stdio.h>
int main()
{
int a = 1, b = 2;
scanf("a=%d", &a);
scanf("b=%d", &b);
printf("a=%d, b=%d\n", a, b);
return 0;
}
输入示例:
a=99↙
a=99, b=2
输入a=99,按下回车键,程序竟然运行结束了,只有个 scanf() 成功读取了数据,第二个 scanf() 仿佛没有执行一样,根本没有给用户任何机会去输入数据。如果我们换一种输入方式呢?
a=99b=200↙
a=99, b=200
这样 a 和 b 都能够正确读取了。注意,a=99b=200中间是没有任何空格的。肯定有好奇的小又问了,如果
a=99b=200两个数据之间有空格又会怎么样呢?我们不妨亲试一下:a=99 b=200↙
a=99, b=2
你看,第二个 scanf() 又读取失败了!在前面的例子中,输入的两份数据之前都是有空格的呀,为什么这里不能带空格呢,真是匪夷所思。好吧,这个其实还是跟缓冲区有关系,我将在《结C语言缓冲区谈scanf()函数》中深入讲解。要想破解 scanf() 输入的问题,一定要学习缓冲区,它能使你对输入输出的认识上升到一个更高的层次,以后不管遇到什么疑难杂症,都能迎刃而解。可以说,输入输出的“门”就在于缓冲区。
输入其它数据
除了输入整数,scanf() 还可以输入单个字符、字符串、小数等,请看下面的演示:
#include <stdio.h>
int main()
{
char letter;
int age;
char url[30];
float price;
scanf("%c", &letter);
scanf("%d", &age);
scanf("%s", url); //可以加&也可以不加&
scanf("%f", &price);
printf("26个英文字母的一个是 %c。\n", letter);
printf("C语言中文网已经成立%d年了,网址是 %s,开通VIP会员的价格是%g。\n", age, url, price);
return 0;
}
运行示例:
z↙ #include <stdio.h> int main() { char author[30], lang[30], url[30]; scanf("%s %s", author, lang); printf("author:%s \nlang: %s\n", author, lang); scanf("%s", url); printf("url: %s\n", url); return 0; } 运行示例: YanChangSheng C-Language↙ 格式控制符 说明 %c 读取一个单一的字符
6↙
http://c.biancheng网站站点" rel="nofollow" />
author:YanChangSheng
lang: C-Language
http://c.biancheng网站站点" rel="nofollow" />
%hd、%d、%ld
读取一个十进制整数,并分别赋值给 short、int、long 类型
%ho、%o、%lo
读取一个八进制整数(可带前缀也可不带),并分别赋值给 short、int、long 类型
%hx、%x、%lx
读取一个十六进制整数(可带前缀也可不带),并分别赋值给 short、int、long 类型
%hu、%u、%lu
读取一个无符号整数,并分别赋值给 unsigned short、unsigned int、unsigned long 类型
%f、%lf
读取一个十进制形式的小数,并分别赋值给 float、double 类型
%e、%le
读取一个指数形式的小数,并分别赋值给 float、double 类型
%g、%lg
既可以读取一个十进制形式的小数,也可以读取一个指数形式的小数,并分别赋值给 float、double 类型
%s
读取一个字符串(以空白符为结束)
- 随机文章
- 鄂博 马尔代夫 视频(美丽的鄂博与梦幻的马尔代夫,一段难忘的旅程)
- 马尔代夫婚礼定制(马尔代夫定制婚礼,助您实现梦幻婚礼 - )
- 中国往返马尔代夫(中资航空开通中国与马尔代夫直航航线)
- 去马尔代夫防晒服(马尔代夫出现防晒服,你需要来看看!)
- 佛山马尔代夫攻略(佛山游客必看!马尔代夫旅游攻略分享)
- 中国马尔代夫系列(发现中国南太重要组成部分之马尔代夫)
- 吴刚现身马尔代夫(吴刚现身马尔代夫,亲水拍摄照片曝光)
- 酷航 马尔代夫(酷航新增马尔代夫线路!)
- 嘉峪关去马尔代夫(从嘉峪关到马尔代夫:一段遥远的旅途)
- 户外马尔代夫别墅(宁静美好的马尔代夫户外别墅住宿体验)
- 揭阳惠来马尔代夫(惠来旅游马尔代夫,开启度假闲适生活)
- 宜兴马尔代夫游泳(体验海底世界,畅游宜兴公园水上乐园)
- 柳版马尔代夫地址(柳版马尔代夫住宿推荐,享受浪漫假期)
- 哈尔滨到马尔代夫(从哈尔滨直飞马尔代夫:探索度假胜地)
- 到马尔代夫去旅游(探索马尔代夫:美丽海岛上的旅行指南)
- 形容马尔代夫黄昏(马尔代夫美丽的黄昏,绝对让你惊叹!)
- 郑州 直飞 马尔代夫(直飞马尔代夫的郑州航班火热开售!)
- 德国 马尔代夫(德国游客涌向马尔代夫度假胜地)
- 冲绳 马尔代夫(从冲绳到马尔代夫:热带天堂之旅)
- 海底套房马尔代夫(马尔代夫的海底酒店房间,让您惊叹!)
- 海底潜水马尔代夫(探秘马尔代夫海底:潜水员的奇妙之旅)
- 希腊 马尔代夫(希马两地成为欧洲游客度假新热点)
- 自驾游到马尔代夫(从自驾游到马尔代夫——探索热带风情)
- 只要 马尔代夫(重返天堂,探索马尔代夫的海洋世界!)
- 比马尔代夫更好的(超越马尔代夫:另一片度假胜地的发现)
- 即墨 马尔代夫(即墨将与马尔代夫建立友好关系)
- 烟台马尔代夫长岛(烟台重返马尔代夫!长岛恢复接待游客)
- 类似马尔代夫岛屿(打造独一无二的珊瑚岛屿,美不胜收!)
- 芜湖马尔代夫风景(走进芜湖仿佛来到马尔代夫,美景无限)
- 老高专属马尔代夫(老高的私人度假天堂——马尔代夫岛屿)
