一:概述
在 C 语言中,字符串实际上是使用空字符 \0 结尾的一维字符数组。因此,\0 是用于标记字符串的结束。
空字符(Null character)又称结束符,缩写 NULL,是一个数值为 0 的控制字符,\0 是转义字符,意思是告诉编译器,这不是字符 0,而是空字符。
比如单词“hello”
char greeting[6] = {'h', 'e', 'l', 'l', 'o', ''};
由于在末尾多了个"",所以字符数组的大小比单词"hello"的字符数多一个。
依据数组初始化规则,可以把上面的语句写成以下语句:char greeting[] = "hello";
实例:
#include <stdio.h>
int main(){
char greeting[6] = {'h', 'e', 'l', 'l', 'o', '\0'};
printf("%s \n", greeting);
return 0;
}
结果输出为:hello
二:字符串函数
函数 | 描述 |
---|---|
strcpy(s1, s2); | 复制字符串s2到字符串s1 |
strcat(s1, s2); | 连接字符串s2到字符串s1的末尾 |
strlen(s1); | 返回字符串s1的长度 |
strcmp(s1, s2); | 如果s1和s2是相同的,则返回0;如果s1<s2则返回小于0;如果s1>s2则返回大于0 |
strchr(s1, ch); | 返回一个指针,指向字符串s1中字符ch的第一次出现的位置。 |
strstr(s1, s2); | 返回一个指针,指向字符串s1中字符串s2的第一次出现的位置。 |
实例:
#include <stdio.h>
int main(){
char str1[12] = "hello";
/*
可以 char str1[12] = "hello";
不可以,char str1[12];
str1 = "hello";
这是因为定义了str1数组之后,然后再用等于号给str1赋值,这个=号要求左边是变量,而定义的数组名是常量
*/
char str2[12] = "world";
char str3[12] = "cc";
int len;
strcpy(str1, str3); // 复制str3到str1
printf("strcpy(str3,str1): %s \n", str3);
/*
这里的结果是ccc,为什么呢?
先了解str1[12]="hello";这里面的12指的是这个数组的长度,而单词hello占5个长度,其实后面还跟着\0,一共占用6个长度,
这6个长度合起来称之为字符串长度。在C语言的字符串中,遇到\0也就是NULL代表着这个字符串的结束。
那么,同样的str3这个数组里面的字符串其实是:"cc\0",而str1:"hello\0",
经过strcpy之后,应该变成cc\0lo\0,第一个\0代替了l的位置。但是前面说过,在C语言的字符串中,遇到\0也就是NULL代表着这个字符串的结束。
所以,结果是cc
*/
strcat(str1, str2); // 连接str2到str1的末尾
printf("strcat(str1, str2): %s \n", str1);
len = strlen(str1); // 连接后,str1的总长度,不包含\0
printf("strlen(str1): %d \n", len);
int num;
num = strcmp(str1, str2);
printf("srcmp(str1, str2): %d \n", num); // -1
int ch = 'w';
int *ptr;
ptr = strchr(str1, ch);
printf("%c开始之后的字符串是 %s \n", ch, ptr);
return 0;
}
结果:
┌──(root💀kali)-[~/Desktop/c_test]
└─# ./zifuchuan
strcpy(str3,str1): cc
strcat(str1, str2): ccworld
strlen(str1): 7
srcmp(str1, str2): -20
w开始之后的字符串是 world
strcpy 函数由于不对数组边界进行检查,而非常容易造成各种缓冲区溢出的漏洞。这些漏洞很容易被利用,而造成严重的系统问题。在使用 strcpy 函数时,要小心谨慎。