=========================================================================

相关代码gitee自取:C语言学习日记: 加油努力 (gitee.com)

=========================================================================

接上期

学C的第二十七天【指针的进阶(三)】_高高的胖子的博客-CSDN博客

=========================================================================

前言:

(1).

C语言中对于字符字符串的处理很是频繁,但是C语言本身没有字符串类型字符串通常放在 常量字符串 中或者 字符数组

(2).

字符串常量 适用于那些对其不做修改字符串函数

~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

1 . 函数介绍

(1). strlen()函数:求字符串(字符数组)长度

函数返回值类型和相关参数:

size_t strlen ( const char * str );

(接收 常量字符指针 ,返回 无符号整数

注意事项:

(1).

字符串以’\0′ 作为结束标志,需要 string.h 头文件strlen函数返回的是在字符串中 ‘\0’ 前面出现的字符个数不包 含 ‘\0’ )。

(2).

参数指向的字符串必须要以 ‘\0’ 结束(不然统计个数为随机值)。

(3).

注意函数的返回值为size_t,是无符号的( 易错 ),如下:

(4).

学会strlen函数的模拟实现(下面第2个模块有)


(2). strcpy()函数:拷贝字符串

函数返回值类型和相关参数:

char* strcpy(char * destination, const char * source );

(接收 目标空间指针内容不能被修改的源字符串指针 ,返回 目标空间指针

注意事项:

(1).

源字符串指针 指向的 字符串 复制目标空间指针 指向的空间中,

包括 终止空字符即 \0 ,并在该点停止

(2).

源字符串必须以 ‘\0’ 结束

(3).

会将源字符串中的 ‘\0’ 拷贝到目标空间

(4).

目标空间必须足够大,以确保能存放源字符串

(5).

目标空间必须可变

(6).

学会strcpy函数的模拟实现(下面第2个模块有)


(3). strcat()函数:追加字符串到另一字符串后

函数返回值类型和相关参数:

char * strcat ( char * destination, const char * source );

(接收 目标空间指针内容不能被修改的源字符串指针 ,返回 目标空间指针

注意事项:

(1).

源字符串的副本追加到目标字符串

目的地中的 ‘\0’ 源字符串的第一个字符覆盖

并且 ‘\0’在新字符串的末尾

需要头文件。

(2).

源字符串必须以 ‘\0’ 结束

(3).

目标空间必须有足够的大,能容纳下源字符串的内容

(4).

目标空间必须可修改

(5).

学会strlen函数的模拟实现(下面第2个模块有)


(4). strcmp()函数:比较两个字符串的大小

函数返回值类型和相关参数:

int strcmp ( const char * str1, const char * str2 );

(接收 两个常量字符串的首字符指针,返回 一个整数

注意事项:

(1).

此函数开始比较时会比较每个字符串的第一个字符

如果它们彼此相等,则继续比较下一对

直到 字符不同 比较完所有字符都相同 为止

(2). 标准规定

第一个字符串的字符大于 第二个字符串的字符,则返回 大于0 的数字

(假设 “abcdef” 和 “abc” 比较,abc都相等第四次比较时 e>\0,那么“abcdef”更大返回 大于0 的数字

第一个字符串的字符 等于 第二个字符串的字符,则 返回0

(假设 “abc” 和 “abc” 比较,abc都相等第四次比较时 \0==\0返回 0

第一个字符串的字符 小于 第二个字符串的字符,则返回 小于0 的数字

(假设 “abcdef” 和 “abq” 比较,a和b都相等第三次比较时 c<q,那么“abq”更大返回小于0 的数字

(3).

学会strlen函数的模拟实现(下面第2个模块有)


(5). strncpy()函数:拷贝字符串的num个字符

函数返回值类型和相关参数:

char * strncpy ( char * destination, const char * source, size_t num );

和 strcpy()函数相同,只是参数多了个 拷贝的字符个数

注意事项:

(1).

将 源字符串指针 指向的 字符串前num个字符复制到 目标空间指针 指向的空间中,

包括 终止空字符,即 \0 ,并在该点停止。(和strcpy()函数类似但限制了拷贝个数

(2).

源字符串拷贝num个字符到目标空间

(3).

如果源字符串的长度小于num,则拷贝完源字符串之后,在目标的空间后边追加0直到num个


(6). strncat()函数:追加字符串的num个字符到另一字符串

函数返回值类型和相关参数:

char * strncat ( char * destination, const char * source, size_t num );

和 strcat()函数相同,只是参数多了个 拷贝的字符个数

注意事项:

(1).

源字符串的副本的num个字符追加到目标字符串。

目的地中的 ‘\0’ 被源字符串的第一个字符覆盖,

并且 ‘\0’ 会在新字符串的末尾。

和strcat()函数类似但限制了追加个数

(2).

如果源字符串的长度小于num,也只追加 源字符串 的字符串
不会像strncpy一样补0


(7). strncmp()函数:在num个字符内比较两个字符串

函数返回值类型和相关参数:

int strncmp ( const char * str1, const char * str2, size_t num );

和 strcmp()函数相同,只是参数多了个 限定在num个字符中比较

注意事项:

(1).

比较到出现另个字符不一样或者一个字符串结束或者num个字符全部比较完


(8). strstr()函数:在字符串中找子字符串

函数返回值类型和相关参数:

char * strstr ( const char *str1, const char * str2);

(参数接收 两个常量字符串,返回 字符指针

注意事项:

(1).

str1 中 找 str2 首次出现的地址,如果 str2 中没有 str1 返回 空指针NULL

(4).

学会strstr函数的模拟实现(下面第2个模块有)


(9). strtok()函数:使用自定义分隔符对字符串进行分割

函数返回值类型和相关参数:

char * strtok ( char * str, const char * sep );

(参数接收 被分割的字符串首字符地址 指定的分隔符指针

返回 被分割的子字符串指针

注意事项:

(1).

sep参数是个字符串,定义了用作分隔符的字符集合

(2).

str参数指定一个字符串,它包含了0个或者多个由sep字符串中一个或者多个分隔符分割的标记

(3).

strtok函数找到str中的下一个标记,并将其用 \0 结尾把找到的标记变为 \0 ),

返回一个指向这个标记的指针

(注: strtok函数会改变被操作的字符串,所以在使用strtok函数切分的字符串一般都是临时拷贝的内容 并且可修改。)

(4).

strtok函数第一个参数不为 NULL ,函数将找到str中第一个标记,strtok函数将保存它在字符串中的位置

(5).

strtok函数第一个参数为 NULL ,函数将在同一个字符串中被保存的位置开始查找下一个标记

(6).

如果字符串中不存在更多的标记,则返回 NULL 指针


(10). strerror()函数:将错误码以字符串形式提供出来

函数返回值类型和相关参数:

char * strerror ( int errnum );

(参数接收 错误码,返回 错误码的字符串信息地址

注意事项:

(1).

库函数在执行时,发生了错误,会将一个错误码存放在 errno 这个变量中

errno C语言提供的一个全局变量

~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

2 . 库函数的模拟实现

(1). 模拟实现strlen()函数:

方法1:计数器方式

对应代码:

#include //方法1:计数器方式size_t my_strlen(const char* str)//和库函数的strlen函数一样//返回值为无符号整型,参数为常量字符指针{int count = 0; //计数器while (*str != '\0')//还没到结束符就继续统计{count++;//没到结束符就统计+1str++;//判断下一位}return count;//返回计数器}int main(){size_t sz = my_strlen("abc");//使用模拟实现的strlen函数返回一个size_t(无符号整数)的数//进行打印:printf("%u\n", sz);//%u:打印无符号的数return 0;}

方法2:指针方式

对应代码:

#include //方法2:指针方式size_t my_strlen(const char* str)//和库函数的strlen函数一样//返回值为无符号整型,参数为常量字符指针{char* p = str;while (*p != '\0')//还没到结束符就继续统计{p++;//没指向结束符就判断下一位}return p-str;//用移动后的指针 - 原来的指针 == 字符串长度}int main(){size_t sz = my_strlen("abc");//使用模拟实现的strlen函数返回一个size_t(无符号整数)的数//进行打印:printf("%u\n", sz);//%u:打印无符号的数return 0;}

方法3:迭代方式

对应代码:

#include //方法3:迭代方式size_t my_strlen(const char* str)//和库函数的strlen函数一样//返回值为无符号整型,参数为常量字符指针{if (*str == '\0'){return 0;//如果第一个就指向\0,说明长度是0}else{return 1 + my_strlen(str + 1);}}int main(){size_t sz = my_strlen("abc");//使用模拟实现的strlen函数返回一个size_t(无符号整数)的数//进行打印:printf("%u\n", sz);//%u:打印无符号的数return 0;}

(2). 模拟实现strcpy()函数:

主函数:

模拟实现strcpy()函数:

对应代码:

//模拟strcpy函数:#include #include char* my_strcpy(char* dest, const char* src)//返回值类型 和 参数 与库函数strcpy相同//返回值设置为char*,是为了能够使用链式访问,把返回值作为其它函数的参数{//保存目标空间指针原位置,方便最后进行返回char* ret = dest;//进行断言,两个指针都不为空指针,需要头文件assert(dest != NULL);assert(src != NULL);//只要 源字符串 还没指向 \0 就继续循环拷贝while (*src != '\0'){*dest = *src; //将源字符串的一位赋给目标空间//下次赋值下一位,所以要移动两个指针:dest++;src++;}//循环中没有将 \0 赋给目标空间,所以要加上:*dest = *src;//循环完后,dest移向了新位置,src在\0位置,所以直接赋值即可//返回拷贝后的目标指针原地址:return ret;}int main(){//目标空间(字符数组):char arr1[20] = "xxxxxxxxxxxxxxxx";//源字符串:char arr2[] = "hello world";my_strcpy(arr1, arr2); //使用模拟的函数来拷贝//arr1为目标空间指针,arr2为源字符串指针//将arr2指向的内容 拷贝到 目标空间指针中//打印拷贝结果:printf("%s\n", arr1);return 0;}

可进行化简:

对应代码:

//模拟strcpy函数:#include #include char* my_strcpy(char* dest, const char* src)//返回值类型 和 参数 与库函数strcpy相同//返回值设置为char*,是为了能够使用链式访问,把返回值作为其它函数的参数{//保存目标空间指针原位置,方便最后进行返回char* ret = dest;//进行断言,两个指针都不为空指针,需要头文件assert(dest != NULL);assert(src != NULL);//只要 源字符串 还没指向 \0 就继续循环拷贝while (*dest++ = *src++){;}//返回拷贝后的目标指针原地址:return ret;}int main(){//目标空间(字符数组):char arr1[20] = "xxxxxxxxxxxxxxxx";//源字符串:char arr2[] = "hello world";my_strcpy(arr1, arr2); //使用模拟的函数来拷贝//arr1为目标空间指针,arr2为源字符串指针//将arr2指向的内容 拷贝到 目标空间指针中//打印拷贝结果:printf("%s\n", arr1);return 0;}

(3). 模拟实现strcat()函数:

对应代码:

//模拟strcat函数:#include #include #include char* my_strcat(char* dest, const char* src){//进行断言,两个字符串都不为\0 (空)assert(dest);assert(src);//保存目标地址的原位置char* ret = dest;//找到目标字符串的\0,作为连接的起点:while (*dest){dest++;}//开始连接字符串,和strcpy是一样的while (*dest++ = *src++){;}//返回连接后的目标空间指针:return ret;}int main(){//目标空间(数组):char arr1[20] = "hello";//源字符串:char arr2[] = " world";//使用模拟的自定义函数,将arr2连接到arr1后my_strcat(arr1, arr2);//打印连接后的新字符串:printf("%s\n", arr1);return 0;}

(4). 模拟实现strcmp()函数:

对应代码:

//模拟strcmp函数:#include int my_strcmp(const char* str1, const char* str2){while (*str1 == *str2)//两字符串同位置上的值相同的情况{if (*str1 == '\0')//同位置上都是\0说明两个字符串相同{return 0; //相同则返回 0}//不是\0,是其它值相同,则判断下一位str1++;str2++;}if (*str1 > *str2)//当前位置,字符串1的字符大于字符串的字符{return 1; //大于则返回大于0的数}else//当前位置,字符串1的字符小于字符串的字符{return -1; //小于则返回小于0的数}}int main(){//使用自定义函数进行比较:int ret = my_strcmp("abq", "abc");printf("%d\n", ret);return 0;}

(5). 模拟实现strstr()函数:

模拟的自定义函数:

主函数:

对应代码:

//模拟strncmp函数:#include #include char* my_strstr(char* str1, char* str2){char* cp = str1; //开始进行判断的初始位置指针char* s1 = cp; //在arr1中的cp位置开始逐位进行匹配的指针char* s2 = str2; //在arr2中逐位进行匹配的指针//如果要找的子字符串为空指针,则返回str1:if (*str2 == '\0'){return str1;}while (*cp != '\0')// \0之后不可能找到arr2的内容{//开始匹配:s1 = cp; //让s1在初始位置开始进行逐位判断s2 = str2;while (*s1 != '\0' && *s2 != '\0' && *s1 == *s2)//两字符串都未到\0,如果当前位置的内容相同,则再循环判断下一位{s1++;s2++;}if (*s2 == '\0')//如果子字符串已经到了\0,//说明arr1中有arr2,匹配成功{return cp; //匹配成功,则返回子字符串的初始位置}cp++; //判断arr1下一个初始位置}return NULL; //未找到则返回空指针}int main(){char arr1[] = "abbbcdef";char arr2[] = "bbc";//使用自定义函数进行查找:char* ret = my_strstr(arr1, arr2);if (ret != NULL){printf("%s\n", ret);}else{printf("未找到");}return 0;}