printf()、sprintf()和scanf()函数为输入/输出函数,也简称为I/O函数,函数声明包含在/usr/include/stdio.h头文件里面。
printf()函数:该函数的返回值为打印字符的数目。
格式:printf(“Control-string”, item1, item2,...)
控制字符串(Control-string)是一个描述如何打印的字符串,用双引号包含。item1,item2是要打印的内容,可以是变量,也可以是常量,也可以是进行计算的表达式。
printf()函数什么时候真正把输出传送给屏幕?首先printf()函数将输出传递给一个被称为缓冲区(buffer)的中介存储区域。缓冲区的内容再不断地传递给屏幕。当缓冲区满、遇到换行符以及需要输入的时候,就会将缓冲区的数据传递给屏幕或者文件。这个过程称为刷新缓冲区(flushing buffer)
转换说明符(conversion specification)及打印输出的结果
转换说明 | 输出结果 |
---|---|
%a | 浮点数、十六进制数和p-记数法(C99) |
%A | 浮点数、十六进制数和p-记数法(C99) |
%c | 一个字符 |
%d | 有符号十进制整数 |
%e | 浮点数、e-记数法 |
%E | 浮点数、E-记数法 |
%f | 浮点数、十进制记数法 |
%g | 根据数值不同自动选择%f或%e。%e格式在指数小于-4或者大于等于精度时使用 |
%G | 根据数值不同自动选择%f或%E。%E格式在指数小于-4或者大于等于精度时使用 |
%i | 有符号十进制数(与%d相同) |
%o | 无符号八进制整数 |
%p | 指针 |
%s | 字符串 |
%u | 无符号十进制整数 |
%x | 使用十六进制数字0f的无符号十六进制整数 |
%X | 使用十六进制数字0F的无符号十六进制整数 |
%% | 打印一个百分号 |
例子:
printf("This is %s.\n", “testing”);
printf()函数的转换说明修饰符:可以在%和定义的转义字符之间通过插入修饰符对基本的转换说明加以修改。修饰符 | 意义 |
---|---|
标志 | 五种标志(-,+,空格,#he 0)将在下表中说明,可以使用零个或者多个标志 |
digit(s) | 字段宽度的最小值,如果该字段不能容纳要打印的数或字符串,系统就会使用更宽的字段 |
.digit(s) | 精度。对于%e、%E和%f转换,是将要在小数点的右边打印的数字的位数。对于%g和%G转换,是有效数字的最大位数。对于%s转换,是将要打印的字符的最大数目。对于整数转换,是将要打印的数字的最小位数,如果必要,要使用前导零来达到这个位数。e.g.”%5.2f“打印字段宽度为5个字符,小数点后有两个数字。 |
h | 和整数转换说明符一起使用,表示一个short int或者unsigned short int类型数值。e.g.“%hu" |
hh | 和整数转换说明符一起使用,表示一个signed char或者unsigned char类型数值 |
j | 和整数转换说明符一起使用,表示一个intmax_t或者uintmax_t值 |
l | 和整数转换说明符一起使用,表示一个long int或者unsigned long int类型值 |
ll | 和整数转换说明符一起使用,表示一个long long int或者unsigned long long int类型值 (C99) |
L | 和浮点数转换说明符一起使用,表示一个long double值 |
t | 和整数转换说明符一起使用,表示一个ptrdiff_t值(与两个指针之间的差相对应的类型) (C99) |
z | 和整数转换说明符一起使用,表示一个size_t值(sizeof返回的类型) (C99) |
标志 | 意义 |
---|---|
- | 项目是左对齐,也就是说,会把项目打印在字段的左侧开始处。e.g.”%-20s” |
+ | 有符号的值若为正,则显示带加号的符号;若为负,则带减号的符号。e.g.“%+6.2f“ |
(空格) | 有符号的值若为正,则显示时带前导空格(但是不显示符号);若为负,则带减号符号。+标志会覆盖空格标志。e.g.”% 6.2f" |
# | 使用转换说明的可选形式。若为%o格式,则以0开始;若为%x和%X格式,则以0x或0X开始,对于所有的浮点形式,#保证了即使不跟任何数字,也打印一个小数点符号。对于%g或%G格式,它防止尾随零被删除。e.g.“%#o“ |
0 | 对于说有的数字格式,用前导零而不是用空格填充字段宽度。如果出现-标志或者指定了精度(对于整数)则忽略该标志。e.g.”%010d“ |
sprintf()函数:它的作用和printf()函数一样,但是它是把输出写到字符串里面,而不是输出显示。
例如:
sprintf(my_strings, "this is testing."); //其中my_strings是用来存放字符串的,可以是字符数组,和字符指针
scanf()函数:格式与printf()函数一样,只是printf()函数打印的项目可以是变量名、常量和表达式,而scanf()函数使用指向变量的指针。scanf()函数的返回值为成功读入的项目的个数。简单规则:
- 如果使用scanf()函数来读取某种基本变量类型的值,要在变量名前面加上取地址符(&)
- 如果使用scanf()函数把一个字符串读进一个字符数组,直接用数组名就可以了
ANSI C中scanf()的转换说明符
转换说明符 | 意义 |
---|---|
%c | 把输入解释成一个字符 |
%d | 把输入解释成一个有符号十进制整数 |
%e,%f,%g,%a | 把输入解释成一个浮点数(%a是C99标准) |
%E,%F,%G,%A | 把输入解释成一个浮点数(%A是C99标准) |
%i | 把输入解释成一个有符号十进制整数 |
%o | 把输入解释成一个有符号八进制整数 |
%p | 把输入解释成一个指针(一个地址) |
%s | 把输入解释成一个字符串:输入的内容以第一个非空白字符作为开始,并且包含直到下一个空白字符的全部字符 |
%u | 把输入解释成一个无符号十进制整数 |
%x,%X | 把输入解释成一个有符号十六进制整数 |
修饰符 | 意义 |
---|---|
* | 滞后赋值。e.g.”%*d“ |
digit(s) | 最大字段宽度:在达到醉倒字段宽度或者遇到第一个空白字符时(不管哪个先遇到)停止对输入项的读取。e.g."%10s" |
hh | 把整数读作signed char或unsigned char e.g.“%hhd” “%hhu” |
ll | 把整数读作long long或者unsigned long long(C99) e.g.“%lld” “%llu” |
h,l或L | “%hd”和“%hi”指示该值会存储在一个short int中。“%ho”、“%hx"和”%hu“指示该值将会存储在一个unsigned short int中。”%ld“和”%li“指示该值将会存储子啊一个long中。”%lo“、”%lx“和”%lu“指示该值将会存储在一个unsigned long中。”%le“、”%lf"he "%lg“指示该值以double类型存储。将L(而非l)与e、f和g一起使用指示该值以long double类型存储。如果没有这些修饰符,d、i、o和x指示int类型,而e、f和g指示float类型。 |
例子:
[root@localhost c]$ cat test.c
#include <stdio.h>
void main(void)
{
char a[10];
int i=0, j=0;
printf("Please input one string and one integer: ");
scanf("%s %d", a, &i); //一般输入,两个格式说明符中间没有其他字符(这里是逗号)
printf("The input string is %s, and the input integer is %d.\n", a, i);
printf("Please input two integers: ");//两个格式说明符中间有其他字符(这里是逗号),在输入的时候,必须要把逗号也作为输入的一部分scanf("%d, %d", &i, &j);
printf("The first input integer is %d, and the second input integer is %d.\n", i, j);
//两个格式说明符中间有其他字符(这里是逗号),在输入的时候,尽管输入的时候吧逗号作为输入的一部分,还是有问题
printf("Please input one string and one integer: ");
scanf("%s, %d", a, &i);
printf("The input string is %s, and the input integer is %d.\n", a, i);
}
编译&执行:[root@localhost c]$ gcc test.c
[root@localhost c]$ ./a.out
Please input one string and one integer: testing 4 //正常输入,用空格作为分隔符
The input string is testing, and the input integer is 4.
Please input two integers: 4,5 //因为个格式说明符中间有逗号,输入的时候也要输入逗号
The first input integer is 4, and the second input integer is 5.
Please input one string and one integer: hello,6 //尽管输入了逗号,但是还是有问题
The input string is hello,6, and the input integer is 4.
printf()函数和scanf()函数的*修饰符
printf()函数与*修饰符,如果不想事先指定字段宽度,而是希望由程序来指定该值,可以再字段宽度部分使用*代替数字。
[root@localhost c]$ cat test.c
#include <stdio.h>
void main(void)
{
unsigned width, precision;
int number = 256;
double weight = 242.5;
printf("What field width?\n");
scanf("%d", &width);
printf("The number is:%*d: \n", width, number); //用输入的字段宽度来打印number变量
printf("Now enter a width and a precision: \n");
scanf("%d %d", &width, &precision);
printf("Weight = %*.*f\n", width, precision, weight);
}
[root@localhost c]$ gcc test.c //编译
[root@localhost c]$ ./a.out //执行
What field width?
6
The number is: 256: //6个字符宽度
Now enter a width and a precision:
8 3
Weight = 242.500 //8个字符宽度,小数点后面三位
而scanf()函数与*修饰符同printf()函数不一样,当把*修饰符放到%和说明符字母之间时,跳过相应的输入项目。[root@localhost c]$ cat test.c
#include <stdio.h>
void main(void)
{
int n;
printf("Please enter three integers.\n");
scanf("%*d %*d %d", &n); //直接跳过前面两个输入项目
printf("The last integer is %d.\n", n);
}
[root@localhost c]$ gcc test.c
[root@localhost c]$ ./a.out
Please enter three integers.
1 2 3
The last integer is 3.