C语言_格式化输入输出
我们知道在最开始输入输出的printf和scanf里可以用百分号跟上一个字母d 说明要整数的类型,%f说明要以浮点数输出
printf()
%[flags][width][.prec][hlL]type
scanf()
%[flag]type
我们知道在最开始输入输出的printf和scanf里可以用百分号跟上一个字母d 说明要整数的类型,%f说明要以浮点数输出,殊不知这其中还有很多数值....
%[flags][width][.prec][hlL]type
type | 含义 |
---|---|
- | 左对齐 |
+ | 在前面放+或- |
(space) | 正数留空 |
0 | 0填充 |
我们来试一下这样的事情,如下 做一个简单的模型 |
printf("%9d\n",123);
printf("%-9d\n",123);
第一个%9d表示后面数字的输出占据九个字符的空间,并且是右对齐的,所以前面留出来空格 而第二个带-号表示是左对齐
"+" 号含义是后面要跟上+号或者- 号,我们试试
printf("%+9d\n",123);
printf("%+-9d\n",123);
运行结果: 果不其然 ,前面加上了+号
printf("%+9d\n",-123);
printf("%+-9d\n",-123);
运行结果: 这样都没什么问题 其实如果不要第二行这个+ 号,编译器也同样会加上-号的,那么我们加0呢
printf("%09d\n",123);
printf("%0-9d\n",-123);
运行结果: 可以看到右对齐的前面补了零,而左对齐的-123是不能补零的,总不能向后填零吧 |width或prec | 含义 | |--|--| | number |最小字符数| ||下一个参数是字符数| |.number|小数点后的位数| |.|下一个参数是小数点后的位数
printf("%9.2f\n",123.0);
运行结果: 这里注意一下, 小数点前面的9指的是整个数字包括小数点以及小数点后面的0,总和占9个字符 而不是小数点前面占9个字符,.2f 表示精确到小数点后俩位 那么那个*又是表达什么意思呢
printf("%*d\n",5,123);
printf("%9.2f\n",123.0);
运行结果: 可以看到第一行后面的形参多了一个,就好比把5 这个值替换了*,而后面的123是用来满足这个%d的,原本要填入%后面的数字,现在可以替换出来,而一旦变成后面的参数,那他就可以是一个变量了,而不是常量,给我们的代码格式带来了很大的灵活性
类型修饰 | 含义 |
---|---|
hh | 单个字节 |
h | short |
l | long |
ll | long long |
L | long double |
type | 作用 |
---|---|
i或d | int |
u | unsigned int |
a或A | 十进制浮点数 |
0 | 八进制 |
c | char |
x | 十六进制 |
s | 字符串 |
X | 字母大写的十六进制 |
p | 指针 |
f或F | float |
n | 读写/写入的个数 |
e或E | 指数 |
暂时列出来,有些学到后面会慢慢遇见 ,不过这里要讲一下n这个type |
int num;
printf("%dsos%n\n",12345,&num);
运行结果: 字符串后面的第一个参数12345,满足了%d,后面的sos也输出了, %n说明符输出有效字符数量,然后把个数传入后面的指针里。
- scanf
- %[flag]tpye
flag | 含义 |
---|---|
* | 跳过 |
数字 | 最大字符数 |
hh | char |
h | short |
l | long ldouble |
ll | long long |
L | long double |
有一点我们需要注意,我们要确保转换说明的数量、类型、和后面的参数的数量、类型一样,例如下面的语句中有俩个%d转换说明,说明后面还有俩个参数
printf("%d 个苹果 %d 元\n",number,money);
后面的确有俩个参数 用来满足俩个%d转换说明 现在C语言通过函数原型机制检查函数调用时参数的个书和类型是否正确,但是这对printf()和scanf()函数不起作用,原因是因为这个俩个函数的参数个数可变,如下:
int a=4;
int b=6;
float x=3.14f;
float y=8.0f;
printf("%d\n",a,b); //参数过多
printf("%d %d\n",a); //参数过少
printf("%d %d\n",x,y); //转换说明类型不匹配
运行结果: 用%d显示float类型的值,这个值不会被转成int类型,在不同的平台下面,运行结果也会导致不同
注意:通常3.14和8.0都会被编译器存储在64位的double类型中,使用双精度进行乘法运算,然后将乘积截断成float的快读,这样虽然计算精度更高,但无疑会减慢运行速度,我们不是在做非常精确的计算 ,所以不用那么精准,可以在浮点数后面加上F或f后缀就可以覆盖编译器默认的是指,编译器会将浮点类型常量看作float类型 ** 没有后缀的浮点常量是double类型。** 很显然这都不是我们想要的结果,所有的编译器都能编译并且运行这个程序,但是大部分的编译器都会给予警告,的确,有些编译器会捕捉到这类问题,但是c标准没有对此多要求,所以,计算机在运行时可能不会捕捉到这类错误,如果程序正常运行,很难察觉出来,大概只有程序没有打印出你所期望的或者意想不到的值的时候,你才会去检查prinf()函数中的参数类型和个数是否正确