C语言学习(二)——字符串和格式化输入输出
1.char数组类型和空字符
C没有为字符串定义专门的变量类型,而是把它存储在char数组里。数组的最后一个位置显示字符\0。这个字符就是空字符,C用它来标记字符串的结束,其ASCII码的值为(或者等同于)0。C的字符串存储时通常以这个空字符结束,该字符的存在意味着数组的单元数必须至少比要存储的字符数多1。计算机可以自己处理大多数这些细节问题(例如,scanf( )会添加‘\0‘使得数组内容成为C字符串)。
2.strlen( )函数与sizeof运算符
sizeof运算符以字节为单位给出数据的大小,strlen( )函数以字符为单位给出字符串的长度。
#include<stdio.h> #include<string.h> /*提供strlen( )函数的原型*/ #define PRAISE "What a super marvelous name!" int main(void) { char name[40]; printf("What‘s your name?\n"); scanf("%s",name); printf("Hello, %s, %s\n", name, PRAISE); printf("Your name of %d letters occupies %d memory cells.\n", strlen(name), sizeof(name)); printf("The phrase has %d letters", strlen(PRAISE)); printf("and occuples %d memory cells .\n", sizeof(PRAISE)); return 0; }
输出结果:
What‘s your name?
Morgan Buttercup
Hello, Morgan. What a super marvelous name;
Your name of 6 letters occupies 40 memory cells.
The phrase of praise has 28 letters and occupies 29 memory cells.
根据sizeof运算符的报告,数组name有40个内存单元。不过只用了其中前6个单元来存放Morgan,这是strlen( )所报告的。数组name的第七个单元中放置空字符,它的存在告诉strlen( )在哪里停止计数。
对于PRAISE,你会发现strlen( )再一次给出了字符串中字符(包括空格和标点符号)的准确数目。Sizeof运算符提供给您的数目大1,这是因为它把用来标志字符串结束的不可见的空字符也计算在内。您并没有告诉计算机为存储该语句分配多大内存,它必须自己计算出双引号之间的字符的数目。
3.常量
1)预处理器允许定义常量,只需在程序文件的顶部添加如下信息即可:
#define TAXRATE 0.015
当编译你的程序时,值0.015将会在TAXRATE出现的每个地方代替它。
2)使用const关键字把一个变量声明转化成常量声明:
const int MOUTHS = 12;
4.printf( )和scanf( )
printf( )函数也有一个返回值,它返回所打印的字符数目;
scanf的返回值有后面的参数决定:
scanf("%d%d", &a, &b);
如果a和b都被成功读入,那么scanf的返回值就是2
如果只有a被成功读入,返回值为1
如果a和b都未被成功读入,返回值为0
如果遇到错误或遇到end of file,返回值为EOF。
且返回值为int型.
如果使用scanf( )来读取某种变量类型的值,请在变量名之前加上&。
如果使用scanf( )把字符串读进一个字符数组,请不要在变量名之前加上&。
要打印%,应该使用%%。
空白字符包括空格,制表符,换行符。C使用空白字符分隔各个语言符号:scanf( )使用空白字符分隔相邻的输入项。
格式转换说明符printf() / scanf()
表一 转换说明符及作为结果的打印输出
转换说明 |
输 出 |
%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() 修饰符
修饰符 |
意 义 |
标志 |
五种标志 (-、+、空格、# 和0) 都将在表三中描述,可以使用零个或多个标志 |
digit(s) |
字段宽度的最小值。如果该字段不能容纳要打印的数或者字符串,系统会使用更宽的字段。示例:“%4d” |
.digit(s) |
精度。对于%e、%E和%f转换,是将要在小数点的右边打印的数字的位数。对于%g和%G转换,是有效数字的最大位数。对于%s转换,是将要打印的字符的最大数目。对于整数转换,是将要打印的数字的最小位数;如果必要,要使用前导零来达到这个位数。只使用“.”表示其后跟随一个零,所以%.f与%.0f相同。示例:“%5.2f”打印一个浮点数,他的字段宽度为5个字符,小数点后有两个数字。 |
h |
和整数转换说明符一起使用,表示一个short int 或者 unsigned short int 类型数值。 示例:“%hu”、“%hx”和“%6.4hd” |
hh |
和整数转换说明符一起使用,表示一个signed char 或者unsigned char类型数值。 示例:“%hhu”、“%hhx”和“%6.4hhd” |
j |
和整数转换说明符一起使用,表示一个intmax_t或uintmax_t值。 示例:“%jd”和“%8jX” |
l |
和整数说明符一起使用,表示一个long int 或者unsigned long int 类型值。 示例:“%ld”和“%8lu” |
ll |
和整数说明符一起使用,表示一个long long int或 unsigned long long int 类型值 (C99)。 示例:“%lld”和“%8llu” |
L |
和浮点转换说明符一起使用,表示一个long double值。 示例:“%Lf”和“%10.4Le” |
t |
和整数转换说明符一起使用,表示一个ptrdiff_t值(与两个指针之间的差相对应的类型) (C99) 示例:“%td”和“%12ti” |
z |
和整数转换说明符一起使用,表示一个size_t值(sizeof返回的类型) (C99)。 示例:“%zd”和“%12zx” |
表三 printf()的标志
修饰符 |
意 义 |
- |
项目是左对齐的,也就是说,会把项目打印在字段的左侧开始处。示例:“%-20s” |
+ |
有符号的值若为正,则显示带加号的符号;若为负,则带减号的符号。示例:“%+6.2f” |
(空格) |
有符号的值若为正,则显示时带前导空格(但是不显示符号);若为负,则带减号符号。+标志会覆盖空格标志。示例:“% 6.2f” |
# |
使用转换说明的可选形式。若为%o格式,则以0开始;若为%x和%X格式,则以0x或0X开始,对于所有的浮点形式,#保证了即使不限任何数字,也打印一个小数点字符。对于%g和%G格式,它防止尾随零被删除。示例:“%#o”、“%#8.0f”和“%+#10.3E” |
0 |
对于所有的数字格式,用前导零而不是用空格填充字段宽度。如果出现-标志或者指定了精度(对于整数)则忽略该标志。示例:“%010d”和“%08.3f” |
表四 ANSIC 中 scanf()的转换说明符
转换说明符 |
意 义 |
%c |
把输入解释成一个字符 |
%d |
把输入解释成一个有符号十进制整数 |
%e,%f,%g, %a |
把输入解释成一个浮点数 (%a是C99标准) |
%E,%F,%G,%A |
把输入解释成一个浮点数 (%A是C99标准) |
%i |
把输入解释成一个有符号十进制整数 |
%o |
把输入解释成一个有符号八进制数 |
%p |
把输入解释成一个指针(地址) |
%s |
把输入解释成一个字符串;输入的内容以一个非空白字符作为开始,并且包含知道下一个空白字符的全部字符 |
%u |
把输入解释成一个无符号十进制整数 |
%x,%X |
把输入解释成一个有符号十六进制整数 |
表五 scanf()的转换修饰符
修饰符 |
意 义 |
* |
滞后赋值。示例:“%*d” |
digit(s) |
最大字段宽度;在达到最大字段宽度或者遇到第一个空白字符时(不管哪一个先发生都一样)停止对输入项的读取。示例:“%10s” |
hh |
把整数读作signed char 或 unsigned char 。示例:“%hhd”“%hhu” |
ll |
把整数读作long long或者 unsigned long long (C99)。示例:“%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”和“%lg”指示该值以double类型存储。将L(而非l)与e、f和g一起使用指示该值以long double类型存储。 如果没有这些修饰符,d、i、o和x指示int类型,而e、f和g指示float类型。 |