打从进入IT行业超过一年。不幸的是,我连printf, scanf都忘了...
一、printf()
首先我们谈谈printf(), 学术化的讲, printf()是格式化输出函数。
printf("<格式化字符串>", <参量表>);
格式化字符串规定两部分内容:
- 正常字符, 按正常输出。
- 格式化规定字符, 以"%"开始,后跟一个或几个规定字符, 用来确定输出内容格式。
格式化字符串的格式是:
%[标志][输出最小宽度][.精度][长度]格式字符
参量表:
需要输出的一系列参数, 个数需要与格式化字符所说名的参数个数一样多, 参数间用","分开, 且顺序对应。
1. 格式化输出
一般标志字符分为四种, - + # 空格。
字符 | 含义 |
---|---|
- | 结果左对齐, 右边填空格 |
+ | 输出符号(正号或负号) |
空格 | 输出值为正时冠以空格, 为负时冠以符号 |
# | 对c, s, d, u类型无影响, 即字符、字符串、十进制有符号整数型、十进制无符号整数型无影响。 o类型, 即无符号八进制类型,输出时前缀为0。 x类型, 即无符号十六进制类型,输出时前缀为0x。 e, f, g类, 即指数型的浮点数类型、浮点数类型、自动选择类型时, 只有结果有小数时才给出。 |
测试函数
void formatPrintfTest() {
int i = 12345;
float j = 678.45;
printf("format = %d\n", i); // 正常输出
printf("format = %10d\n", i); // 位数不足时, 左侧补空格
printf("format = %-10d\n", i); // 位数不足时, 右侧补空格
printf("format = %+d\n", i); // 输出时追加前缀正负号
printf("format = % d\n", i); // 输出时正整数前缀留空格, 负整数时前缀留负号
printf("format = %#o\n", i); // 输出无符号八进制数, 前缀为0
printf("format = %#x\n", i); // 输出无符号十六进制数, 前缀为0x
printf("format = %#f\n", j); // 浮点数输出有小数点时会显示
}
2. 指定宽度输出
输出最小宽度, 使用十进制表示时的最少位数。即少则留空格, 占位。多则正常输出
测试函数:
void widthPrintfTest() {
int i = 123;
printf("width = %2d\n", i); // 实际位数大于定义宽度: 按实际输出
printf("width = %5d\n", i); // 实际位数小于定义宽度: 在左侧补空格
printf("width = %-5d\n", i); // 实际位数小于定义宽度: 负号, 左对齐, 右侧留空格
printf("width = %05d\n", i); // 实际位数小于定义宽度: 表宽度数字从0开始, 则右对齐, 左侧补0
printf("width = %-05d\n", i); // 无实际意义
}
3. 指定长度输出
长度格式符就分h, l, h表示短整型输出, l表示长整型输出
测试函数:
void lengthPrintfTest() {
int h = 12345678901;
int i = 1234567890;
int j = 123456;
int k = 12345;
printf("lenth = %hd\n", j); // 可保证输出正确五位整型数
printf("lenth = %hd\n", k); // 可保证输出正确五位整型数
printf("lenth = %ld\n", i); // 可保证输出正确十位整型数
printf("lenth = %ld\n", h); // 可保证输出正确十位整型数
}
4. 指定精度输出
输出精度, 以"."开头, 跟十进制整数描述。
输出如果是数字, 则表示小数位数; 实际位数大于定义精度, 进行四舍五入, 否则追加位数到精度。
输入如果是字符, 则表示输出字符个数; 实际位数大于定义精度, 进行字符截取。
测试函数:
void accuracyPrintfTest() {
float i = 1234.5678;
char c[10] = "helloworld";
printf("accuracy = %.2f\n", i);
printf("accuracy = %.9f\n", i);
printf("accuracy = %.3s\n", c);
printf("accuracy = %.9s\n", c);
}
二、scanf()
相比较printf()是格式化输出函数, scanf()则是与之对应的格式化输入函数。
scanf("<格式化字符串>", <地址表>);
相类似的格式化字符串包含三类不同字符:
- 格式化说明符, 与printf()中基本相同。
- 空白字符, 空白字符的使用会使scanf()函数在读操作中略去输入中的一个或多个空白字符。
- 非空白字符, 一个非空白字符会使scanf()函数在读操作时剔除与这个非空白字符相同的字符。
地址表:
地址表读取是所有变量的地址, 而不是变量本身。
- 一般变量, 在变量前加上"&", 输出时用变量名。
- 数组, 用数组名就代替了数组首地址, 输出时也用数组名。
- 指针, 用指针名本身, 不要加上"*", 输出时也用指针即可。
1. 格式化输出
scanf()函数中是否包含(非)空白字符, 导致输入格式不同。
测试函数
void blankScanfTest() {
int i, j;
/*
此时如果中间定义了",", 输入的时候需要注意","的输入
用户输入全英文回车, 输出均为0。
用户输入123,456回车 ==> i = 123, j = 456
用户输入123456回车 ==> i = 123456, j = 0
*/
scanf("%d,%d", &i, &j);
printf("i = %d, j = %d\n", i, j);
}
void blankScanfTest() {
int i, j;
/*
并列的数据源输入需要使用空格隔开, 即下列两种写法相同
要么输入123回车, 456回车 ==> i = 123, j = 456
要么输入123空格456 ==> i = 123, j = 456
*/
scanf("%d%d", &i, &j);
// scanf("%d %d", &i, &j);
printf("i = %d, j = %d\n", i, j);
}
2. 更多使用方法
测试函数:
void moreScanfTest() {
char c1, c2;
printf("输出两个字符\n");
/*
如果输入A回车, 因为输入一条语句完毕必须回车, 此时由于回车符仍存在在缓存区,
故执行输入语句scanf(%c, &c2)后, 变量c2输出是一空行, 即A回车 ==> c1 = A, c2 = 空行
如果用户输入AB, 输出c1 = A, c2 = B
用户输入超出两个字符以上的话, 会截取最前面两个字符进行分配
*/
scanf("%c", &c1);
scanf("%c", &c2);
printf("c1 = %c, c2 = %c\n", c1, c2);
}