c语言|C语言中,C进阶中字符函数、字符串函数、内存函数详解。

本内容是重点介绍使用字符函数、字符串函数、内存函数的库函数的使用和注意事项。
前言

求字符串长度字符串查找
strlenstrstr
长度不收限制的字符串函数错误信息报告
strcpystrerror
strcat内存操作函数
strcmpmemcpy
长度受限制的字符串函数介绍memmove
strncpymemset
strncatmemcmp
strncmp字符操作
1.strlen:求字符串长度
size_tstrlen (constchar*str);
注意事项:
  • 字符串已'\0'作为结束标志,strlen函数返回的是在字符串中'\0'前面出现的字符个数(不包含\0)
  • 参数指向的字符串必须要以'\0' 结束
  • 注意函数的返回值是size_t (unsigned int),是无符号的
实例:
#include #includeint main() { char arr1[]= "abcdef"; int ret = strlen(arr1); printf("%d\n", ret); return 0; }

模拟实现strlen函数
#includesize_t my_strlen(const char* str) { char* st = str; while (*st++); return st-str-1; } int main() { char arr1[] = "abcdef"; int ret = my_strlen(arr1); printf("%d\n", ret); }

c语言|C语言中,C进阶中字符函数、字符串函数、内存函数详解。
文章图片


2. strcpy : 字符串拷贝
char*strcpy(char * destination, constchar * source)
注意事项:
  • 源字符串必须以 '\0' 结束。
  • 会将源字符串中的 '\0' 拷贝到目标空间。
  • 目标空间必须足够大,以确保能存放源字符串。
  • 目标空间必须可变。
实例:
#include #include int main() { char arr1[20] = { 0 }; char arr2[] = "abcdef"; printf("%s\n",strcpy(arr1, arr2)); return 0; }

模拟实现strcpy:
#include #include char* my_strcpy(char* dest, const char* src) { assert(dest && src); char* ret = dest; while (*dest++ = *src++) { ; } return ret; }int main() { char arr1[20] = { 0 }; char arr2[10] = "abcdef"; printf("%s\n", my_strcpy(arr1,arr2)); return 0; }


3.strcat :字符串连接(字符串追加)
char * strcat ( char * destination, const char * source)
注意事项:
  • 源字符串必须以 '\0' 结束。
  • 目标空间必须有足够的大,能容纳下源字符串的内容。
  • 目标空间必须可修改。
实例:
#include #include int main() { char a1[20] = "abc"; char a2[] = "def"; printf("%s\n", strcat(a1, a2)); return 0; }

模拟实现strcat:
#include char* my_strcat(char* dest, const char* src) { char* ret = dest; while (*dest) { dest++; } while (*dest++ = *src++) { ; } return ret; } int main() { char a1[20] = "abc"; char a2[] = "def"; printf("%s\n", my_strcat(a1, a2)); }

c语言|C语言中,C进阶中字符函数、字符串函数、内存函数详解。
文章图片

4.strcmp :字符串比较,比较的是字符串的内容,不是长度
int strcmp ( const char * str1,const char * str2);
标准规定:
  • 第一个字符串大于第二个字符串,则返回大于0的数字
  • 第一个字符串等于第二个字符串,则返回0
  • 第一个字符串小于第二个字符串,则返回小于0的数字
实例:
#include #include int main() { char a1[] = "abcd"; char a2[] = "abcd"; int ret = strcmp(a1, a2); printf("%d\n", ret); return 0; }

模拟实现strcmp:
#include int my_strcmp(const char* s1, const char* s2) { while (*s1 == *s2) { if (*s1 == '\0') { return 0; } s1++; s2++; } return *s1 - *s2; } int main() { char a1[] = "abcd"; char a2[] = "abcd"; int ret = my_strcmp(a1, a2); printf("%d\n", ret); return 0; }

5.strncpy
char * strncpy ( char * destination , const char * source , size_t num);
注意事项:
  • 拷贝num个字符从源字符串到目标空间。
  • 如果源字符串的长度小于num,则拷贝完源字符串之后,在目标的后边追加0,直到num个。
实例:
#include #include int main() { char arr1[20] = { 0 }; char arr2[] = "abcdef"; printf("%s\n", strncpy(arr1, arr2 + 2, 2)); //输出结果cd return 0; }

6.strcat
char * strncat ( char * destination , const char * source ,size_t num);
实例:
#include #include int main() { char str1[20]; char str2[20]; strcpy(str1, "To be "); strcpy(str2, "or not to be"); strncat(str1, str2, 6); puts(str1); //输出结果,To be or not return 0; }


7.strncmp
int strncmp ( const char * str1, const char * str2, size_t num );
注意事项: 比较到出现另个字符不一样或者一个字符串结束或者num个字符全部比较完。c语言|C语言中,C进阶中字符函数、字符串函数、内存函数详解。
文章图片

实例:
#include #include int main() { char str[][5] = { "R2D2","C3PO ","R2A6" }; int n; puts("Looking for R2 astromech droids..."); for (n = 0; n < 3; n++) if (strncmp(str[n], "R2xx", 2) == 0) { printf("found %s\n", str[n]); //输出结果R2D2R2A6 } return 0; }

8. strstr :字符串查找函数 在一个字符串中查找子字符串
char * strstr ( const char * str2, const char * str1);
实例:
#include #include int main() { char arr1[] = "i am a good student,hehe student"; char arr2[] = "student"; //查找arr1中arr2第一次出现的位置 char* ret = strstr(arr1, arr2); if (ret == NULL) { printf("找不到\n"); } else { printf("%s\n", ret); //输出 student,hehe student } return 0; }

模拟实现strstr:
#include #include char* my_strstr(const char* str1, const char* str2) { assert(str1 && str2); char* s1; char* s2; char* cp = str1; if (*str2 == '\0') { return str1; } while (*cp) { s1 = cp; s2 = str2; while (*s1 != '\0' && *s2 != '\0' && *s1 == *s2) { s1++; s2++; } if (*s2 == '\0') { return cp; } cp++; } return NULL; }int main() { char a1[] = "hello my name is wz zz"; char a2[] = "wz"; char* ret = my_strstr(a1, a2); if (ret == NULL) { printf("找不到了\n"); } else { printf("%s\n", ret); } return 0; }

9.strtok :分割字符串
char * strtok ( char * str, const char * sep );
注意事项 :
  • sep参数是个字符串,定义了用作分隔符的字符集合
  • 第一个参数指定一个字符串,它包含了0个或者多个由sep字符串中一个或者多个分隔符分割的标 记。
  • strtok函数找到str中的下一个标记,并将其用 \0 结尾,返回一个指向这个标记的指针。(注: strtok函数会改变被操作的字符串,所以在使用strtok函数切分的字符串一般都是临时拷贝的内容 并且可修改。)
  • strtok函数的第一个参数不为 NULL ,函数将找到str中第一个标记,strtok函数将保存它在字符串 中的位置。
  • strtok函数的第一个参数为 NULL ,函数将在同一个字符串中被保存的位置开始,查找下一个标 记。
  • 如果字符串中不存在更多的标记,则返回 NULL 指针。
实例:
#include #include int main() { char arr1[] = "wzzz@bwwwu.errrh"; char arr2[100] = { 0 }; //临时数据 //zpw\0bitedu.tech\0 char sep[] = "@."; strcpy(arr2, arr1); char* ret = NULL; //分割字符串 for (ret=strtok(arr2, sep); ret!=NULL; ret=strtok(NULL, sep)) { printf("%s\n", ret); } //strtok(arr2, sep); //strtok(NULL, sep); return 0; }

10.strerror
char * strerror(int errnum);
返回误码,所对应的错误信息 。
实例:
#include #include int main() { printf("%s\n", strerror(0)); printf("%s\n", strerror(1)); printf("%s\n", strerror(2)); printf("%s\n", strerror(3)); return 0; }

c语言|C语言中,C进阶中字符函数、字符串函数、内存函数详解。
文章图片

11.memcpy :拷贝的整形数据
void * memcpy ( void * destination, const void * source, size_t num );
注意事项:
  • 函数memcpy从source的位置开始向后复制num个字节的数据到destination的内存位置。
  • 这个函数在遇到 '\0' 的时候并不会停下来。
  • 如果source和destination有任何的重叠,复制的结果都是未定义的。
实例:
#include #include struct { char name[40]; int age; } person, person_copy; int main() { char myname[] = "Pierre de Fermat"; memcpy(person.name, myname, strlen(myname) + 1); person.age = 46; memcpy(&person_copy, &person, sizeof(person)); printf("person_copy: %s, %d \n", person_copy.name, person_copy.age); return 0; }


模拟实现memcpy:
#include #include void* my_memcpy(void* dest, const void* src, size_t count) { void* ret = dest; assert(dest && src); while (count--) { *(char*)dest = *(char*)src; dest = (char*)dest+1; src = https://www.it610.com/article/(char*)src+1; } return ret; }int main() { int arr[10] = { 1,2,3,4,5,6,7,8,9,10 }; int i = 0; my_memcpy(arr + 2, arr, 16); for (i = 0; i < 10; i++) { printf("%d ", arr[i]); } return 0; }

12.memmove :内存拷贝时,出现内存重复的现象
void * memmove ( void * destination, const void * source, size_t num );
注意事项:
  • 和memcpy的差别就是memmove函数处理的源内存块和目标内存块是可以重叠的。
  • 如果源空间和目标空间出现重叠,就得使用memmove函数处理。
实例:
#include #include int main () { char str[] = "memmove can be very useful. ........... "; memmove (str+20,str+15,11); puts (str); return 0; }

模拟实现memmove:
#include #include void* my_memmove(void* dest, const void* src, size_t count) { void* ret = dest; assert(dest && src); if (dest < src) { //dest在src左边时,从前往后拷贝 while (count--) { *(char*)dest = *(char*)src; dest = (char*)dest + 1; src = https://www.it610.com/article/(char*)src + 1; } return ret; } else { //dest在src右边时,从后往前拷贝 while (count--) { *((char*)dest+count) = *((char*)src+count); } return ret; } }int main() { int arr[10] = { 1,2,3,4,5,6,7,8,9,10 }; int i = 0; my_memmove(arr + 2, arr, 16); for (i = 0; i < 10; i++) { printf("%d ", arr[i]); } return 0; }

13.memcmp
int memcmp ( const void * ptr1, const void * ptr2, size_t num );
注意事项:
  • 比较从ptr1和ptr2指针开始的num个字节
  • 返回值:
c语言|C语言中,C进阶中字符函数、字符串函数、内存函数详解。
文章图片

实例:
#include #include int main() { int arr1[] = { 1,2,3,4,5 }; int arr2[] = { 1,2,3,6,6 }; int ret = memcmp(arr1, arr2, 12); printf("%d\n", ret); //输出结果为 0 return 0; }

14. memset
void * memset(void *dest,int c,size_t count);
实例:
#include #include int main() { int arr[] = { 1,2,3,4,5,6 }; memset(arr, 1, 20); return 0; }

【c语言|C语言中,C进阶中字符函数、字符串函数、内存函数详解。】

    推荐阅读