随手记录一下比较重要的知识点。
定义字符串:字符串常量、CHAR数组、CHAR指针、字符串数组
字符串常量属于静态储存类。
整个引号中的内容作为指向该字符串储存位置的指针。
例如:*”space fakers” 代表 S
指定字符串大小时,多余的初始化为”\0”
声明一个数组时,数组的大小必须为整型常量,而不能是运行时得到的变量值。(除变长数组)编译时,数组的大小被锁定到程序中。
CHAR数组名的值是数组首元素的地址。
指针和数组的区别
Const char *m3[] = ”\n what’s your name?”
Char m3[] = ”\n what’s your name?”
看起来,这两种赋值形式是相同,其实并不完全相同,这是为什么呢?
数组初始化是从静态储存区复制一个字符串给数组,而指针初始化只是复制字符串的地址。
数组名是地址常量,值为数组的首地址,不可更改,不允许使用自增/自减运算符。(储存在常量储存区?)
而指针名代表的地址是储存在静态储存区内的,属于地址变量,可以改变,允许使用自增/自减运算符。
数组的元素是变量,但数组名是常量。
注:内存的五大分区:堆(NEW)、栈(编译器)、自由储存区(malloc、free)、全局和静态储存区(变量)、常量储存区(常量)
指针的初始化问题
Char * word = ”Frame”;
Word[1] = ‘L’;
这样的语句是允许的吗?答案是否定的
可能编译器一般会将所有相同的字符串指针指向相同的地址。
这样printf(“%s”,”Frame”);会打印出FLame
有的编译器这样工作,有的直接产生程序异常中断。
因此,安全的写法是:
Const char * word = “Frame”;
字符串数组的声明
两种方式,一种是CHAR数组的数组
Char mytal2[m][n];
一种是CHAR指针的数组
Char * mytal[m];
这个不规则数组不浪费任何储存空间
输入字符串
首先,预留字符串的储存空间,然后,通过字符串输入函数。
典型错误:
Char * name;
Scanf(“%s”,name);
name是一个未初始化的地址。
可以:
Char name[81];
或者使用后面介绍的分配储存空间的函数
Gets()函数读取换行符之前(不包括换行符)的所有字符,在这些字符后面添加一个空字符(\0),并丢弃换行符。返回值时字符串的地址。
常见使用方法:
While(gets(name) != NULL)
注:比while(ch = getchar() != EOF)要简洁多了
NULL是一个空指针。区别空指针和空字符。
Gets()的不足:不检查预留存储区是否能够容纳实际输入的数据,多出来的字符简单地溢出到相邻储存区。(UNIX蠕虫病毒)
Fgets()函数解决了这个问题。
Char* fgets(char *str,int n,FILE * stream)
第二个参数代表读入的最大字符数,如果这个参数值为n,最多读入n-1个,到换行符结束(会将换行符储存到字符串里)。(可以用strchr()来定位和删除换行符)
第三个参数代表读哪个文件,从键盘上读数据时,用stdin作为参数(代表standard input),这个标识符在头文件stdio.h中定义。
适用于安全性要求高的编程。
scanf基于单词获取字符串,还可以设置字符串宽度。
输出字符串
Puts(char *)
puts显示字符串时,自动添加一个换行符。遇到空字符停止读取,因此一定要有空字符。
Fputs( char* , 要写的文件(stdout));
fputs不为输出添加换行符
printf也是以字符串的地址作为参数。
四、自定义字符串输入/输出函数
例如:
Void put1(const char * string)
{
While(*string != ‘\0’)
{
Putchar(string ++);
}
}
注:
使用const修饰符,代表这个函数不改变输入的字符串;
++比优先级高;
这是一个自定义字符串处理函数的范例,因为空字符的存在,不必要传递字符串的长度参数。
While(*string != ‘\0’)可以简写为while(*string)
字符串函数
头文件:#include <string.h>
Strlen()函数可以得到字符串的长度。
缩短字符串:
Void fit(char *string , unsigned int size)
{
If(strlen(string ) > size )
{
*(string + size) = ‘\0’;
}
}
Strcat()函数将第二个字符串的一份拷贝添加到第一个字符串的结尾。(第一个字符串发生改变,第二个不变),返回值为第一个参数。
Strcat()函数的缺点:不检查第一个字符串是否能容纳第二个字符串。
Strncat()函数比strcat()多一个参数指明最多允许添加字符串的长度,遇到第N个字符或者遇到空字符停止。
两种合并两个字符串的方法:
(flower是字符串数组,手动检查第一个字符串能否容纳第二个字符串)
if((strlen(addon) + strlen(flower) +1) <= size)
Strcat(flower , addon);
另一种,直接 strncat(flower,addon,size-strlen(flower));
Strcmp()函数比较两个字符串是否相同(即使存放它们的数组长度不同)
区分大小写。
返回值:第一个字符串在ASCII的顺序先于第二个字符串 负数
相等 0
后于 正数
如果字符串的第一个字符相同,就会一直向后查找,直到找到第一不同的,决定返回正数还是返回负数。
单个字符一般不用STRCMP函数比较
Strnmp()函数可以比较到字符串尾,也可以比较到指定参数处。
Strcpy()函数接受两个指针参数,将第二个指针的值复制给第一个指针。
返回值为char*类型,返回第一个参数的地址。
其次,第一个指针可以不指向字符串的开始,只复制字符串的一部分。
Strncpy()多一个参数指明最大可复制字符串。
Strncpy(qwords[i],temp,targsize-1);
Qwords[i][targsize-1] = ‘\0’;
Sprintf()函数,是stdio.h里的
第一个参数是目标字符串的地址,其他和printf一样,用于格式化输出。
Sprintf(formal,”%s,%-19s: $% 6.2f\n”,last,first,price);
Puts(formal);
Size_t类型是sizeof运算符返回的任意类型。(某一种整数类型)
一个小应用。
Char line[80];
Char*find;
Fgets(line,80,stdin);
Find = strchr(line,’/n’);
综合作业:字符串排序。
ctype.h字符函数和字符串
命令行参数
C编译器允许main()没有参数/有两个参数。
第一个参数是命令行中的字符串数,第二个参数是一个指向字符串的指针数组。
允许用引号将多个单词集成到一个字符串内。
把字符串转成数字
Atoi()接受字符串为参数,返回int。
要使用 stdlib.h头文件。
Atof()把字符串转换为double。
Atol()把字符串转换为long int。
Strtol():LONG
Strtoul():UNSIGNED LONG
Strtod():DOUBLE
可以识别并报告字符串中非数字的第一个字符。
Long strtol(const char * nptr , char **endptr , int base)
第一个参数是希望转换的字符串的指针,第二个参数是非数字的第一个字符的指针的地址,第三个参数是基数。
Itoa():整数转字符串
Fltoa():浮点数转字符串