当前位置:   article > 正文

常用的字符函数和字符串函数

字符函数

 

一、分类

  • 求字符串长度:strlen
  • 长度不受限制的字符串函数:strcpy、strcat、strcmp
  • 长度受限制的字符串函数:strncpy、strncat、strncmp
  • 字符串查找:strstr strtok
  • 错误信息报告:strerror
  • 字符操作
  • 内存操作函数:memcpy、memmove、memset、memcmp                                                                                                   

二、详细介绍

1.strlen

①作用:获取一个字符串长度
②用法:size_t strlen( const char *string )
③头文件:<string.h>
④注意:
  a.字符串已经 '\0' 作为结束标志,strlen函数返回的是在字符串中 '\0' 前面出现的字符个数(不包含 '\0' )
  b. 
参数指向的字符串必须要以 '\0' 结束。
  c.
注意函数的返回值为size_t,是无符号的。
⑤模拟实现
a.计数器实现(
每次往后移动一位,计数器count++。直到遇到\0结束。

  1. int my_strlen(char* str)
  2. {
  3. int count = 0;
  4. while (*str++)
  5. {
  6. count++;
  7. }
  8. return count;
  9. }
  10. int main()
  11. {
  12. char arr[] = "abcdef";
  13. int ret = my_strlen(arr);
  14. printf("%d\n", ret);
  15. return 0;
  16. }

b.递归实现(不创建临时变量)
这样说吧,my_strlen("abc")=1+my_strlen("bc")=1+1+my_strlen("c")=1+1+1+my_strlen(" "),将求整个字符串的长度转换为求1+剩下字符串的长度,层层剥开,这就是递归思想。

  • 递归的限制条件是*str == '\0'时返回
  • 每次调用函数是让str+1,越来越接近这个限制条件
  1. int my_strlen(char* str)
  2. {
  3. if (*str == '\0')
  4. {
  5. return 0;
  6. }
  7. else
  8. {
  9. return my_strlen(str + 1) + 1;
  10. }
  11. }
  12. int main()
  13. {
  14. char arr[] = "abcdef";
  15. int ret = my_strlen(arr);
  16. printf("%d\n", ret);
  17. return 0;
  18. }

c.指针-指针
指针和指针相减是元素个数的绝对值。

  1. int my_strlen(char* str)
  2. {
  3. char *p = str;
  4. while (*p != '\0')
  5. {
  6. p++;
  7. }
  8. return p - str;
  9. }
  10. int main()
  11. {
  12. char arr[] = "abcdef";
  13. int ret = my_strlen(arr);
  14. printf("%d\n", ret);
  15. return 0;
  16. }

2.strcpy和strncpy

①作用:拷贝字符串和拷贝字符串有长度限制
②用法:char *strcpy( char *strDestination, const char *strSource )
              char
*strncpy( char *strDest, const char *strSource, size_t count )
③头文件:<string.h>
④注意:
   a.源字符串必须以 '\0' 结束。
   b.会将源字符串中的 '\0' 拷贝到目标空间。
   c.目标空间必须足够大,以确保能存放源字符串。
   d.目标空间必须可变
   A.拷贝count
个字符从源字符串到目标空间。
   B.如果源字符串的长度小于num
,则拷贝完原字符串后,在目标后面追加0,直到count个。
⑤模拟实现

  1. #include <stdio.h>
  2. #include <assert.h>
  3. char* my_strcpy( char* des,const char* src)
  4. {
  5. assert(des != NULL);
  6. assert(src != NULL);
  7. char* ret = des;
  8. while ((*des++ = *src++))
  9. {
  10. ;
  11. }
  12. return ret;
  13. }
  14. int main()
  15. {
  16. char arr[20] = { 0 };
  17. char str[] = "hello world";
  18. char* ret = my_strcpy(arr, str);
  19. printf("%s\n",(char*) ret);
  20. return 0;
  21. }

 

  1. #include <stdio.h>
  2. #include <assert.h>
  3. char* my_strncpy(char* des, const char* src, size_t count)
  4. {
  5. char* ret = des;
  6. assert(des && src);
  7. while (count && (*des++ = *src++)) //拷贝字符串 (拷贝\0 )
  8. {
  9. count--;
  10. }
  11. if (count) //目标后面追加0
  12. while (--count) //遇到\0就要结束所以为--count
  13. {
  14. *des++ = '\0';
  15. }
  16. return ret;
  17. }
  18. int main()
  19. {
  20. char arr[20] = { 0 };
  21. char str[] = "hello world";
  22. char* ret = my_strncpy(arr, str, 5);
  23. printf("%s\n", ret);
  24. return 0;
  25. }

3.strcat和strncat
①作用:追加字符串和追加字符串长度受限制
②用法:char *strcat( char *strDestination, const char *strSource )
               char *strncat( char *strDest, const char *strSource, size_t count )
③头文件:<string.h>
④注意:
   a.源字符串必须以 '\0' 结束。
   b. 目标空间必须有足够的大,能容纳下源字符串的内容。
   c. 目标空间必须可修改。
   d. 字符串自己给自己追加,造成\0被覆盖,死循环导致数组越界

⑤模拟实现

  1. #include <stdio.h>
  2. #include <assert.h>
  3. char* my_strcat(char* des, const char* src)
  4. {
  5. char* ret = des;
  6. assert(des != NULL);
  7. assert(src != NULL);
  8. //判断\0
  9. while (*des++)
  10. {
  11. des++; //不要把\0传进去
  12. }
  13. //进行拷贝
  14. while ((*des++ = *src++))
  15. {
  16. ; //需要把\0传进去
  17. }
  18. return ret;
  19. }
  20. int main()
  21. {
  22. char arr[20] = "hello ";
  23. char str[] = "world";
  24. char* ret = my_strcat(arr, str);
  25. printf("%s\n", ret);
  26. return 0;
  27. }

 

  1. #include <stdio.h>
  2. #include <assert.h>
  3. char* my_strncat(char* des, char* src, size_t count)
  4. {
  5. char* ret = des;
  6. assert(des && src);
  7. while (*(++des))
  8. {
  9. ;
  10. }
  11. while (count && (*des++ = *src++))
  12. {
  13. count--;
  14. }
  15. *des = '\0';
  16. return ret;
  17. }
  18. int main()
  19. {
  20. char str1[20] = "hello ";
  21. char str2[5] = "world";
  22. char* ret = my_strncat(str1,str2,3);
  23. printf("%s\n", ret);
  24. return 0;
  25. }

4.strcmp和strncmp
①作用:比较两个字符串的大小和比较两个字符串大小长度受限制
②用法:int strcmp( const char *string1, const char *string2 )
             int strncmp( const char *string1, const char *string2, size_t count )
③头文件:<string.h>
④注意:
   第一个字符串大于第二个字符串,则返回大于0的数字
   第一个字符串等于第二个字符串,则返回0
   第一个字符串小于第二个字符串,则返回小于0
的数字
⑤模拟实现

  1. #include <stdio.h>
  2. #include <assert.h>
  3. int my_strcmp(const char* arr,const char* str)
  4. {
  5. int ret = 0;
  6. assert(arr && str);
  7. while (*arr == *str)
  8. {
  9. arr++;
  10. str++;
  11. }
  12. return ret = *arr - *str;
  13. }
  14. int main()
  15. {
  16. char arr[] = "abcd";
  17. char str[] = "abce";
  18. int ret = my_strcmp(arr, str);
  19. if (ret > 0)
  20. printf("大于\n");
  21. else if (ret < 0)
  22. printf("小于\n");
  23. else
  24. printf("等于\n");
  25. return 0;
  26. }

 

  1. #include <stdio.h>
  2. #include <assert.h>
  3. int my_strncmp(char* s1, char* s2, size_t count)
  4. {
  5. int ret = 0;
  6. assert(s1 && s2);
  7. while ((*s1 == *s2) && count)
  8. {
  9. s1++;
  10. s2++;
  11. count--;
  12. }
  13. return ret = *s1 - *s2;
  14. }
  15. int main()
  16. {
  17. char str1[] = "abcef";
  18. char str2[] = "abcdg";
  19. int ret = my_strncmp(str1, str2,3);
  20. if (ret > 0)
  21. printf("大于\n");
  22. else if (ret < 0)
  23. printf("小于\n");
  24. else
  25. printf("等于\n");
  26. return 0;
  27. }

*4.strstr
①作用:在字符串中查找字符串
②用法:char *strstr( const char *string, const char *strCharSet )
③头文件:<string.h>
④注意:
    返回的是被查找字符第一次出现查找字符的首元素地址
⑤模拟实现
 

  1. #include <stdio.h>
  2. #include <assert.h>
  3. char* my_strstr(const char* s1, const char* s2)
  4. {
  5. assert(s1 && s2);
  6. const char* p1 = s1;
  7. const char* p2 = s2;
  8. const char* start = p1;
  9. if (*s2 == '\0')
  10. return s1;
  11. while (*start)
  12. {
  13. p2 = s2;
  14. p1 = start;
  15. while (*p1 == *p2 && *p1 && *p2)
  16. {
  17. p1++;
  18. p2++;
  19. }
  20. if (*p2 == '\0')
  21. {
  22. return (char*)start;
  23. }
  24. start++;
  25. }
  26. return NULL;
  27. }
  28. int main()
  29. {
  30. char str1[] = "abbbcdef";
  31. char str2 [] = "bbc";
  32. char* ret = my_strstr(str1, str2);
  33. if (ret == NULL)
  34. {
  35. printf("找不到\n");
  36. }
  37. else
  38. {
  39. printf("找到了,下标是%s\n", ret);
  40. }
  41. return 0;
  42. }

5.strtok

①作用:切割字符串
②用法:char *strtok( char *strToken, const char *strDelimit )
③头文件:<string.h>
④注意:
a.strDelimit参数是个字符串,定义了用作分隔符的字符集合
b.第一个参数指定一个字符串,它包含了0个或者多个由strDelimit字符串中一个或者多个分隔符分割的标记
c.strtok
函数找到str中的下一个标记,并将其用 \0 结尾,返回一个指向这个标记的指针。(注:strtok函数会改
变被操作的字符串,所以在使用
strtok函数切分的字符串一般都是临时拷贝的内容并且可修改。)
d.strtok
函数的第一个参数不为 NULL ,函数将找到str中第一个标记,strtok函数将保存它在字符串中的位置。
strtok
函数的第一个参数为 NULL ,函数将在同一个字符串中被保存的位置开始,查找下一个标记。
e.如果字符串中不存在更多的标记,则返回 NULL 指针

6.strerror
①作用:返回错误码所对应的错误信息
②用法:char *strerror( int errnum )
③头文件:<string.h>

7.字符分类函数

iscntrl    任何控制字符

isspace    空白字符:空格‘ ’,换页‘\f’,换行'\n',回车‘\r’,制表符'\t'或者垂直制表符'\v'

isdigit    十进制数字 0~9

isxdigit    十六进制数字,包括所有十进制数字,小写字母a~f,大写字母A~F

islower    小写字母a~z

isupper    大写字母A~Z

isalpha    字母a~zA~Z

isalnum   字母或者数字,a~z,A~Z,0~9

ispunct    标点符号,任何不属于数字或者字母的图形字符(可打印)

isgraph    任何图形字符

isprint    任何可打印字符,包括图形字符和空白字符

8.memcpy

①作用:内存拷贝
②用法:void *memcpy( void *dest, const void *src, size_t count )
③头文件:<string.h>
④注意:
a.函数memcpysrc的位置开始向后复制num个字节的数据到dest的内存位置。
b.这个函数在遇到 '\0'
的时候并不会停下来。
c.如果source
destination有任何的重叠,复制的结果就是未定义的。
⑤模拟实现
 

  1. #include <stdio.h>
  2. #include <assert.h>
  3. void* my_memcpy(void* dest, const void* src, size_t count)
  4. {
  5. void* ret = dest;
  6. assert(dest && src);
  7. while(count--)
  8. {
  9. *(char*)dest = *(char*)src;
  10. ++(char*)dest;
  11. ++(char*)src;
  12. //dest = (char*)dest+1;
  13. //src = (char*)src+1;
  14. }
  15. return ret;
  16. }
  17. int main()
  18. {
  19. int arr1[10] = {0};
  20. int arr2[] = {1,2,3,4,5,6 };
  21. my_memcpy(arr1, arr2, 20);
  22. return 0;
  23. }

*9.memmove

①作用:内存移动
②用法:void *memmove( void *dest, const void *src, size_t count )
③头文件:<string.h>
④注意:
a.和memcpy的差别就是memmove函数处理的源内存块和目标内存块是可以重叠的。
b.如果源空间和目标空间出现重叠,就得使用memmove
函数处理。
⑤模拟实现
 

  1. #include <stdio.h>
  2. #include <assert.h>
  3. void* my_memmove(void* des, void* src, size_t count)
  4. {
  5. assert(des && src);
  6. if (src < des) //从后向前
  7. {
  8. while (count--)
  9. {
  10. *((char*)des+count) = *((char*)src+count);
  11. }
  12. }
  13. else //从前向后
  14. {
  15. while (count--)
  16. {
  17. *((char*)des) = *((char*)(src));
  18. ((char*)des)++;
  19. ((char*)src)++;
  20. }
  21. }
  22. }
  23. int main()
  24. {
  25. int arr[] = {1,2,3,4,5,6,7,8,9,10};
  26. my_memmove(arr+2, arr, 16);
  27. return 0;
  28. }

10.memcpy
①作用:内存比较
②用法:void *memcpy( void *dest, const void *src, size_t count )
③头文件:<string.h>
④注意:
比较从dest和src指针开始的count个字节
⑤模拟实现

10.memset
①作用:内存初始化
②用法:void *memset( void *dest, int c, size_t count )
③头文件:<string.h>
④注意:
dest为初始化字符串首元素地址,c为要初始化成的字符,count为初始化字符串大小,单位是字节
 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/小蓝xlanll/article/detail/71463
推荐阅读
相关标签
  

闽ICP备14008679号