#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,按下回车键,scanf() 就会读取输入数据并赋值给变量 a;本次输入结束,接着执行下一个 scanf() 函数,再从键盘输入 60,按下回车键,就会将 60 赋值给变量 b,都是同样的道理。%d
,后面还跟着两个变量,这要求我们一次性输入两个整数,并分别赋值给 c 和 d。注意"%d %d"
之间是有空格的,所以输入数据时也要有空格。对于 scanf(),输入数据的格式要和控制字符串的格式保持一致。scanf("%d %d", &a, &b); // 获取用户输入的两个整数,分别赋值给变量 a 和 b printf("%d %d", a, b); // 将变量 a 和 b 的值在显示器上输出它们都有格式控制字符串,都有变量列表。不同的是,scanf 的变量前要带一个
&
符号。&
称为取地址符,也就是获取变量在内存中的地址。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; }输出结果:
%p
是一个新的格式控制符,它表示以十六进制的形式(带小写的前缀)输出数据的地址。如果写作%P
,那么十六进制的前缀也将变成大写形式。注意:这里看到的地址都是假的,是虚拟地址,并不等于数据在物理内存中的地址。虚拟地址是现代计算机因内存管理的需要才提出的概念,我们将在《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
"%d %d"
,中间有一个空格,而我们却输入了10 20
,中间有多个空格。第二个 scanf() 的格式控制字符串为"%d %d"
,中间有多个空格,而我们却输入了100 200
,中间只有一个空格。这说明 scanf() 对输入数据之间的空格的处理比较宽松,并不要求空格数严格对应,多几个少几个无所谓,只要有空格就行。"%d, %d, %d"
,中间以逗号分隔,所以输入的整数也要以逗号分隔。is bigger than
分隔。
12 60 10 23↙
a+b=72
c*d=230
12 60 10 23 99↙
a+b=72
c*d=230
12 60 10↙
a+b=72
23↙
c*d=230
#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
#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=99b=200
中间是没有任何空格的。a=99b=200
两个数据之间有空格又会怎么样呢?我们不妨亲试一下:
a=99 b=200↙
a=99, b=2
#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↙
6↙
http://c.biancheng.net↙
159.9↙
26个英文字母的最后一个是 z。
C语言中文网已经成立6年了,网址是 http://c.biancheng.net,开通VIP会员的价格是159.9。
char str1[] = "http://c.biancheng.net";
char *str2 = "C语言中文网";
[ ]
里面要指明字符串的最大长度,如果不指明,也可以根据=
后面的字符串来自动推算,此处,就是根据"http://c.biancheng.net"
的长度来推算的。但是在前一个例子中,开始我们只是定义了一个字符串,并没有立即给它赋值,所以没法自动推算,只能手动指明最大长度,这也就是为什么一定要写作char url[30]
,而不能写作char url[]
的原因。&
取地址符,这很容易理解;但是对于此处的 url 字符串,我们并没有加 &,这是因为,字符串的名字会自动转换为字符串的地址,所以不用再多此一举加 & 了。当然,你也可以加上,这样虽然不会导致错误,但是编译器会产生警告,至于为什么,我们将会在《数组和指针绝不等价,数组是另外一种类型》《数组到底在什么时候会转换为指针》中讲解。#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↙
author:YanChangSheng
lang: C-Language
http://c.biancheng.net http://biancheng.net↙
url: http://c.biancheng.net
格式控制符 | 说明 |
---|---|
%c | 读取一个单一的字符 |
%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 | 读取一个字符串(以空白符为结束) |
Copyright © 广州京杭网络科技有限公司 2005-2024 版权所有 粤ICP备16019765号
广州京杭网络科技有限公司 版权所有