一、printf
1.基本用法
printf() 的作用是将参数文本输出到屏幕。它名字里面的f代表 format (格式化),表示可以定制输出文本的格式。作为库函数,使用前需要包含头文件<stdio.h>,函数的返回值是打印字符串的长度。
#include<stdio.h>//头文件
int main()
{
printf("hehe\n");
return 0;
}
运行结果:
此处光标移到了下一行。
#include<stdio.h>
int main()
{
int A = 43;
printf("%d\t",printf("%d\t",printf("%d\t",A)));
//第一次printf打印出来的是43
//第二次printf打印出来的是第一次printf的返回值2+1
//第三次printf打印出来的是第二次printf的返回值1+1
return 0;
}
运行结果:
\t也是一种字符,故要+1。
printf() 不会在行尾自动添加换行符,运行结束后,光标就停留在输出结束的地放,不会自动换 行。 为了让光标移到下一行的开头,可以在输出文本的结尾,添加⼀个换行符 \n 。如若让光标空开一个tab键,则可以添加一个制表符\t。
2.占位符
1.所谓“占位符”,就是这个位置可以用其他值代入。
输出文本里可以有多个占位符。
举个例子:
#include<stdio.h>
int main()
{
printf("%s says it is %d o'clock\n", "lisi", 21);
return 0;
}
这里%s和%d就是“lisi”和“21”的占位符。
2. printf() 参数与占位符是一一对应关系,如果有 n 个占位符, printf() 的参数就应该有 n + 1 个。如果参数个数少于对应的占位符, printf() 可能会输出内存中的任意值。
上面那个里子中:占位符有2个,参数有3个(三段文字就是三个参数)
3.输出格式
①限定宽度
#include<stdio.h>
int main()
{
printf("%5d\n", 123); // 输出为 " 123"
return 0;
}
%5d表示占位符的宽度至少为五位。如果不满五位,默认右对齐,前面会补齐空格,输出的值默认向右对齐。如果超过五位,则正常打印。如果要改成向左对齐,需要在%后加上一个“-”。
#include <stdio.h>
int main()
{
printf("%-5d\n", 123); // 输出为 "123 "
return 0;
}
对于小数,这个限定符会限制所有数字的最小显示宽度。
// 输出 " 123.450000"
#include <stdio.h>
int main()
{
printf("%12f\n", 123.45);
return 0;
}
上面示例中, %12f 表示输出的浮点数最少要占据12位。由于小数的默认显示精度是小数点后6位, 所以 123.45 输出结果的头部会添加2个空格。
②显示符号
printf()输出时只显示“-”而不显示“+”,若要打印的数字显示符号,则需在%后面加上一个“+”
③限定小数位数
若要限制小数打印的位数,则可以在%f中间加上“.n”
若要同时限制最小宽度和小数位数,则可以写成“%6.2f”
在0.50的前面会有俩个空格,来补齐六位。
还有一种方法利用*代替最小宽度和小数位数
④输出部分字符串
%s 占位符用来输出字符串,默认是全部输出。如果只想输出开头的部分,可以用%.[m]s 指定输出 的长度,其中 [m] 代表⼀个数字,表示所要输出的长度。注意:%s是从首元素地址打印到"\0"。
由于“.5”的限制,我们只打印出了字符串的前五个字符。
二、scnaf
1.基本用法
当我们有了变量,我们需要给变量输⼊值就可以使用scanf 函数,如果需要将变量的值输出在屏幕上 的时候可以使用 prinf 函数,下面看⼀个例子:
①注释:
- scanf同printf一样需要占位符,以此知道输入数据的类型
- 同printf文件一样,其头文件为<stdio.h>
- 输入数据后,需要按下回车,数据才会被读取
- 变量前面必须加上 & 运算符(指针变量除外),因为 scanf() 传递的不是值,而是地址, 即将变量 i 的地址指向用户输入的值。 如果这里的变量是指针变量(比如字符串变量),那就不用加 & 运算符
- scanf() 处理数值占位符时,会自动过滤空白字符,包括空格、制表符(\t)、换行符(\n)等。 所以用户输入的数据之间,有一个或多个空格不影响 scanf() 解读数据。另外,用户使用回车键,将输入分成几行,也不影响解读。
②原理:
用户的输入先放入缓存,等到按下回车键后,按照占位符对缓存进行解读。
下面是图解:
2.占位符
scanf() 常⽤的占位符如下:与 printf() 的占位符基本⼀致。
- • %c :字符。
- • %d :整数。
- • %f : float 类型浮点数。
- • %lf : double 类型浮点数。
- • %Lf : long double 类型浮点数。
- • %s :字符串。
- • %[] :在方括号中指定⼀组匹配的字符(比如 %[0-9] ),遇到不在集合之中的字符,匹配将会停止。
上⾯所有占位符之中,除了 %c 以外,都会自动忽略起首的空白字符。 %c 不忽略空白字符,总是返回当前第一个字符,无论该字符是否为空格。如果要强制跳过字符前的空白字符,可以写成 scanf(" %c", &ch) ,即 %c 前加上一个空格,表示跳过零个或多个空白字符。
下面要特别说一下占位符%s ,它其实不能简单地等同于字符串。它的规则是,从当前第一个非空白字符开始读起,直到遇到空白字符(即空格、换行符、制表符等)为止。因为 %s 不会包含空白字符,所以无法用来读取多个单词,除非多个%s 一起使用。这也意味着scanf() 不适合读取可能包含空格的字符串,比如书名或歌曲名(可用gets函数替代)。另外,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个字符,后面的字符将被丢弃,这样就不会有数组溢出的风险了。
3其他问题
①返回值
scanf() 的返回值是一个整数,表示成功读取的变量个数。 如果没有读取任何项,或者匹配失败,则返回 0 。如果在成功读取任何数据之前,发生了读取错误或者遇到读取到文件结尾,则返回常量EOF(-1)。EOF——end of file文件结束标志。
在VS环境中按3次 ctrl+z ,才结束了输入,我们可以看到r是2,表示正确读取了2个数值。 如果一个数字都不输入,直接按3次 ctrl+z ,输出的r是-1,也就是EOF。
②返回值忽略
scanf() 中有一个赋值忽略符(assignment suppression character)“ * ”。 只要把*加在任何占位符的百分号后面,该占位符就不会返回值,解析后将被丢弃。
三、占位符大全
• %a :十六进制浮点数,字母输出为小写。
• %A :十六进制浮点数,字母输出为大写。
• %c :字符。//char
• %d :十进制整数。//int
• %e :使⽤科学计数法的浮点数,指数部分的 e 为小写。
• %E :使⽤科学计数法的浮点数,指数部分的 E 为大写。
• %i :整数,基本等同于 %d 。
• %f :⼩数(包含 float 类型和 double 类型)。//float- %f double- %lf
• %g :6个有效数字的浮点数。整数部分⼀旦超过6位,就会⾃动转为科学计数法,指数部分的 e 为⼩写。
• %G :等同于 %g ,唯⼀的区别是指数部分的 E 为大写。
• %hd :十进制shortint类型。
• %ho :八进制shortint类型。
• %hx :十六进制shortint类型。
• %hu :unsignedshortint类型。
• %ld :十进制longint类型。
• %lo :八进制longint类型。
• %lx :十六进制longint类型。
• %lu :unsignedlongint类型。
• %lld :十进制longlongint类型。
• %llo :八进制longlongint类型。
• %llx :十六进制longlongint类型。
• %llu :unsignedlonglongint类型。
• %Le :科学计数法表⽰的longdouble类型浮点数。
• %Lf :longdouble类型浮点数。
• %n :已输出的字符串数量。该占位符本⾝不输出,只将值存储在指定变量之中。
• %o :八进制整数。
• %p :指针(用来打印地址)。
• %s :字符串。
• %u :无符号整数(unsignedint)。
• %x :十六进制整数。
• %zd : size_t 类型。
• %% :输出一个百分号。