• 【C语言】进阶——字符串和内存函数


     

    目录

    一:非限制字符串函数

    1.strlen

     👊模拟实现

    方法1:计算器法

    方法2.指针-指针

    方法3.函数调用

    2.strcpy

     👊模拟实现

     3.strcat

    👊模拟实现 

    4.strcmp

    👊模拟实现 

    二:可限制字符串函数

    1.strncpy

    2.strncat

    3.strncmp

    4.strstr

     👊模拟实现

    5.strtok

    6.strerror-perror

    三:内存函数 

    1.memcpy

    👊模拟实现

    内存拷贝加强版

     2.memcmp

     3.memset


     

    一:非限制字符串函数

    1.strlen

    size_t strlen ( const char * str )

    求字符串长度;

    头文件:#include

    注:计算遇到\0之前的长度;

            参数所指向的字符串必须以\0结束;

    返回值:返回的是无字符的大小

    1. #include
    2. int main()
    3. {
    4. char* str = "abcdef";
    5. int sl = strlen(str);
    6. printf("%d\n", sl); //6
    7. return 0;
    8. }

     👊模拟实现

    方法1:计算器
    1. #include
    2. size_t my_strlen(char* str)
    3. {
    4. size_t ret;
    5. while(*str)
    6. {
    7. str++;
    8. ret++;
    9. }
    10. return ret;
    11. }
    12. int main() {
    13. char arr[]={"abcdef"};
    14. printf("%zd\n",my_strlen(arr));
    15. return 0;
    16. }
    方法2.指针-指针
    1. #include
    2. size_t my_strlen(char* str)
    3. {
    4. char* ret = str;
    5. while(*str)
    6. {
    7. str++;
    8. }
    9. return str-ret;
    10. }
    11. int main() {
    12. char arr[]={"abcdef"};
    13. printf("%zd\n",my_strlen(arr));
    14. return 0;
    15. }

    指针-指针的绝对值是之间相差的元素个数 

    方法3.函数调用
    1. #include <stdio.h>
    2. size_t my_strlen(char* str)
    3. {
    4. if(*str =='\0')
    5. {
    6. return 0;
    7. }
    8. else
    9. {
    10. return 1+my_strlen(str+1);
    11. }
    12. }
    13. int main() {
    14. char arr[]={"abcdef"};
    15. printf("%zd\n",my_strlen(arr));
    16. return 0;
    17. }

    2.strcpy

    char* strncpy(char* dest, const char* src, size_t n);

    拷贝字符串;

    头文件#include

    将一个字符串的内容拷贝到另一个字符串;

    它的参数都是两个指针,

    第一个参数为目标空间的起始位置(拷贝的所在位置),

    第二个参数是源字符串内容的起始位置,即被拷贝的字符串。

    返回值:返回值是目标空间的起始位置;

    注:

    • 将源指向的字符串复制到目标指向的空间,包括终结空字符
    • 源字符串必须以 ‘\0’ 结束。
    • 会将源字符串中的 ‘\0’ 拷贝到目标空间。
    • 目标空间必须足够大,以确保能存放源字符串。
    • 目标空间必须可变。
    1. #include <string.h>
    2. int main()
    3. {
    4. char arr1[10] = "";
    5. char* arr2 = "abcdef";
    6. strcpy(arr1, arr2);
    7. printf("%s\n", string); //abcdef
    8. return 0;
    9. }

     👊模拟实现

    1. #include<stdio.h>
    2. #include<string.h>
    3. #include<assert.h>
    4. char* my_strcpy(char* dest,const char* src)
    5. {
    6. char* ret = dest; //因为返回的是目标地址,先记录目标起始地址;
    7. assert(dest && src);
    8. while(*dest++ = *src++) //循环 直到遇到\0
    9. {
    10. ;
    11. }
    12. return ret;
    13. }
    14. int main()
    15. {
    16. char temp[] = "xxxxxxxxxxxxxx";
    17. char arr[] = "who say!!!";
    18. char* s = my_strcpy(temp,arr);
    19. printf("%s", s);
    20. return 0;
    21. }

     3.strcat

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

    字符串追加;

    头文件:#include

    strcat函数将源字符串追加到目标字符串的后面;

    它的两个参数是两个指针

            第一个指针指向的是目标字符串的起始位置

            第二个指针指向的是源字符串的起始位置

    返回值:目标空间的起始位置;

    注:

    • 源字符串必须以 ‘\0’ 结束;
    • 目标空间必须有足够的大,能容纳下源字符串的内容;
    • 目标空间必须可修改;
    • 从源字符中\0位置开始追加;
    1. int main()
    2. {
    3. char arr1[20] = "abcdef" ;
    4. char arr2[] = "abdd" ;
    5. printf("%s", strcat(arr1, arr2));
    6. return 0;
    7. }

    👊模拟实现 

    先循环源字符找到'\0',然后解引用追加;

    1. char * my_strcat( char *dest, char *src)
    2. {
    3. char* ret = dest; //记录起始地址
    4. while (*dest) //循环找到/0位置
    5. {
    6. dest++;
    7. }
    8. while (*dest++ = *src++) //从dest \0处开始追加
    9. {
    10. ;
    11. }
    12. return ret; //返回源字符起始地址
    13. }
    14. int main()
    15. {
    16. char arr1[20] = {"abcdefg"};
    17. char arr2[] = " ggb"; //GG爆
    18. printf("%s\n", strcat(arr2, arr2));
    19. }

    4.strcmp

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

    用于比较两个字符串内容的函数;

    头文件:#include

    它的两个参数都是指针,两个指针分别指向待比较的起始位置,

    返回值:

    当str1大于str2的时候返回一个大于0的数;

    小于则返回一个小于0的数;

    等于则返回0;

    注:

    字符串比较的是两个字符串对应的ASCII值而不是字符串的长度。

    1. int main()
    2. {
    3. char arr[] = "abcd";
    4. char arr2[] = "abcdef";
    5. printf("%d", strcmp(arr, arr2)); //返回小于0的数
    6. return 0;
    7. }

    👊模拟实现 

     从两个字符串的起始位置开始比较,

    如果相同且不为‘\0’,则继续比较下一对字符;

    如果相同且等于’\0’直接返回0;

    如果目标空间的字符ASCII值大于源字符串中字符的ASCII值则返回1,否则返回-1。

    1. #include<assert.h>
    2. int my_strcmp(char *str1,char *str2)
    3. {
    4. assert(str1 && str2);
    5. while (*str1==*str2)
    6. {
    7. if (*str1== '\0')
    8. {
    9. return 0;
    10. }
    11. str1++;
    12. str2++;
    13. }
    14. if (*str1> *str2)
    15. {
    16. return 1;
    17. }
    18. else {
    19. return -1;
    20. }
    21. }
    22. int main()
    23. {
    24. char arr1[20] = "abcde";
    25. char arr2[] = "abcdef";
    26. printf("%d", my_strcmp(arr1, arr2));
    27. return 0;
    28. }

    二:可限制字符串函数

    可限制字符串函数,就是可以限制字符串大小的函数;

    1.strncpy

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

    strcpy相比:多了个无字符型num;

    限制几个拷贝字符个数,也会补\0; 

    头文件:#include

    返回值:返回目标字符串首地址;

    注:

    1. 如果源字符串的长度小于num,则拷贝完源字符串之后,在目标的后边追加0,直到num个。
    2. 拷贝num个字符从源字符串到目标空间。num为多少就拷贝多少。
    1. #include<stdio.h>
    2. #include<string.h>
    3. int main()
    4. {
    5. char arr1[] = "xxxxxxxxxxxxxx";
    6. char arr2[] = "who say!!!";
    7. printf("%s", strncpy(arr1, arr2, 10));
    8. return 0;
    9. }

    将arr2中的10个字符拷贝到arr1中;        如果arr2中不足10个字符,直接补‘\0’,

    2.strncat

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

    与strcat相比:多了个size_t型参数;指定操作字符的个数

    在目标字符\0处追加 源字符的num个字符;

    头文件:#include

    返回值:返回目标字符串首地址;

    注:

    1. 如果追加时操作数大于源字符串中字符个数,只会将源字符串的字符个数全部追加完就结束
    2. 追加时是从目标空间的‘\0’开始追加,追加完后再补一个‘\0’,使之成为字符串
    3. 可以自己给自己追加
    1. int main()
    2. {
    3. char arr1[10] = "abcd\0xxxxx";
    4. char arr2[] = "efghg";
    5. printf("%s", strncat(arr1, arr2, 2));
    6. return 0;
    7. }

    3.strncmp

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

    与上同理:num表示源字符限制元素个数比较

    头文件:#include

    返回值:比较规则和strcmp一致;

    1. int main()
    2. {
    3. char arr1[10] = "abc";
    4. char arr2[] = "defg";
    5. printf("%d", strncmp(arr1, arr2, 2));
    6. return 0;
    7. }

    4.strstr

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

    strstr函数可以在一个字符串1中查找另一个字符串2;

    如果字符串2能在字符串1中找到,返回字符串2在字符串1中出现的起始位置;

    否则就返回空指针;

    头文件:#include

    注:

    如果字符串2为空字符串,则返回的是字符串1的起始位置。

    1. int main()
    2. {
    3. char arr1 [] = "abcabc";
    4. char arr2[] = "abc";
    5. char* ret = strstr(arr1, arr2);
    6. if (ret != NULL)
    7. {
    8. printf("找到了");
    9. }
    10. else
    11. {
    12. printf("没有找到");
    13. }
    14. return 0;
    15. }

     👊模拟实现

    1. //模拟ststr()
    2. char* my_strstr(const char* str1, const char* str2)
    3. {
    4. char* cp = str1; //记录相匹配的位置
    5. char* s1 = str1; //遍历字符串str1
    6. char* s2 = str2; //遍历字符串str2
    7. while (s2)
    8. {
    9. while (*s1 == *s2)
    10. {
    11. s1++;
    12. s2++;
    13. }
    14. //s2遍历完了,返回记录相匹配的位置
    15. if (*s2 =='\0')
    16. return cp;
    17. //匹配位置不对,匹配位置+1
    18. cp++;
    19. s1 = cp; //将s1拉回位置
    20. s2= str2; //将s2拉回首地址,重新遍历
    21. //cp遍历完str1,没有相匹配位置 ,返回空指针
    22. if (*cp == '\0')
    23. return NULL;
    24. }
    25. //str2为空,返回str1
    26. return str1;
    27. }

     

    5.strtok

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

    字符分割函数 

    sretok函数通过给定的分隔符的字符集合中的字符去把字符串分割成若干个子字符串

    注意如果分隔符的字符集合中的字符不是待分割字符串的字符,是无法分割。

    所以分割符的字符集合必须是字符串中有的字符。

    它的第一参数是需要被分割的字符串的首地址,第二个参数也是字符串的首地址(不分先后)

    该字符串是作分割符的字符集合。返回值是查找到分隔符之前字符串的首地址。

    头文件:#include

    注:

    1. strtok函数找到str中的下一个标记,并将其用 \0 结尾,返回一个指向这个标记的指针。
    2. strtok函数的第一个参数不为 NULL ,函数将找到str中第一个标记,strtok函数将保存它在字符串 中的位置。
    3. strtok函数的第一个参数为 NULL ,函数将在同一个字符串中被保存的位置开始,查找下一个标记。
    4. 如果字符串中不存在更多的标记,则返回 NULL 指针。
    5. strtok函数会改变被操作的字符串,所以在使用strtok函数切分的字符串一般都是临时拷贝的内容 并且可修改。
    1. //strtok
    2. //char* strtok(char* arr,const char* del)
    3. int main()
    4. {
    5. char arr[] = { "whosay@lana.haha" };
    6. char* p = "@.";
    7. char* ret = strtok(arr, p);
    8. printf("%s\n", ret);
    9. ret = strtok(NULL, p);
    10. printf("%s\n", ret);
    11. return 0;
    12. }

     这样就将代码写死了,因为有时候我们无法确定输入的字符串;

    而strtok函数,可以记住上次置NULL的地址。可以利用循环优化;

    1. int main()
    2. {
    3. char arr[] = { "whosay@.lana.haha" };
    4. char buf[100] = { 0 };
    5. char* p = "@.";
    6. char* s;
    7. for (s = strtok(buf, p); s != NULL; s = strtok(NULL, p))
    8. {
    9. printf("%s\n", s);
    10. }
    11. return 0;
    12. }

    6.strerror-perror

    报错函数 

    返回错误码,所对应的错误信息。

    perror == printf + strerror

    1. int main()
    2. {
    3. FILE* pf = fopen("add.txt", "r");
    4. if (pf == NULL)
    5. {
    6. perror("打开文件失败");
    7. return 1;
    8. }
    9. else
    10. {
    11. printf("打开文件成功\n");
    12. //...
    13. }
    14. return 0;
    15. }

    这两个函数都是将错误码转换成人们能看的懂的信息,

    区别就是strerror是将错误码转换成错误信息后不打印;

    而perror会打印

    而且perror中的字符内容是自己指定;

    三:内存函数 

    1.memcpy

    void * memcpy ( void * destination, const void * source, size_t num );

    mencpy函数是拷贝两块无关的内存区域数据的函数,它会从源数据中的起始位置拷贝num个字节的数据到目标空间里去;

    返回值:返回目标空间的首地址

    注:

    • 函数memcpy从source的位置开始向后复制num个字节的数据到destination的内存位置。
    • 这个函数在遇到 ‘\0’ 的时候并不会停下来。
    • 如果source和destination有任何的重叠,复制的结果都是未定义的。
    1. #include<stdio.h>
    2. int main()
    3. {
    4. int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
    5. int arr2[20] = { 0 };
    6. int* ret = (int*)memcpy(arr2, arr1, 20); //返回值的类型为void*,所以强转。
    7. for (int i = 0; i < 5; i++) {
    8. printf("%d ", *(ret+i));
    9. }
    10. return 0;
    11. }

    一个整型4个字节,要拷贝5个数,所以传参的第三个参数为20.

    👊模拟实现

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

    2.memmove

    内存拷贝加强版

    menmvoe函数和menecpy函数的参数和返回值是一模一样的;

    memmvoe函数和memcpy函数最大的区别:

    1.         memmove函数操作的源内存块和目标空间的内存块是可以重叠的
    2.         而memcpy函数的源内存块和目标空间的内存块是不能重叠
    1. #include<stdio.h>
    2. #include<string.h>
    3. int main()
    4. {
    5. int arr[] = { 1,2,3,4,5,6,7,8,9,10 };
    6. int* ret = (int*)memmove(arr+3 , arr, 20);
    7. for (int i = 0; i <=6; i++) {
    8. printf("%d ", *(ret + i));
    9. }
    10. return 0;
    11. }

     👊模拟实现

    将拷贝情况分为三种

    1. dest指针位于sour内存块的左边,从前向后拷贝。
    2. dest指针在sour内存块内,则从后向前拷贝。
    3. dest指针与sour内存块位于同一区域,则可以从后往前拷贝也可以从前往后拷。

    为了方便分为两种情况,直接划分从前往后拷和从后往前拷。

    1. #include<assert.h>
    2. void* my_memmove(void* dest, const void* sour, size_t count)
    3. {
    4. assert(dest && sour);
    5. void* ret = dest;
    6. if (dest < sour)
    7. {
    8. while (count--)
    9. {
    10. *(char*)dest = *(char*)(sour);
    11. dest = (char*)dest + 1;
    12. sour = (char*)sour + 1;
    13. }
    14. }
    15. else
    16. {
    17. while (count--)
    18. {
    19. *((char*)dest + count) = *((char*)sour + count);
    20. }
    21. }
    22. return ret;
    23. }
    24. int main()
    25. {
    26. int arr1[10] = { 1,2,3,4,5,6,7,8,9,10 };
    27. my_memmove(arr1 + 2, arr1, 16);
    28. int i = 0;
    29. int sz = sizeof(arr1) / sizeof(arr1[0]);
    30. for (i = 0; i < sz; i++)
    31. {
    32. printf("%d ", arr1[i]);
    33. }
    34. return 0;
    35. }

     2.memcmp

    int memcmp ( const void * ptr1,  const void * ptr2,  size_t num );
    

    mencmp函数是比较两个内存块大小的函数

    它会比较ptr1和ptr2开始的num个字节,

    1. 当ptr1>ptr2时,返回一个大于0的数;
    2. 当ptr1
    3. 相等时则返回0;
    1. #include<stdio.h>
    2. #include<string.h>
    3. int main()
    4. {
    5. int arr1[] = { 1,2,3,4,5 };
    6. int arr2[] = { 1,2,3,4,0x1122334455 };
    7. int ret = memcmp(arr1, arr2, 16);
    8. printf("%d\n", ret);
    9. return 0;
    10. }

     3.memset

    1. void *menmset(void* dest ,int num,size_t count)

    内存设置

    memset函数可以将内存块中的的某一部分修改为指定的字符。

    三个参数,

    第一个参数是目标的起始位置,

    第二个参数是指定的修改内存区域的字符,

    第三个参数是从起始位置开始设置的内存的字节个数。

    memset是以字节为单位来初始化内存单元的。

    1. int main()
    2. {
    3. char arr[] = "hello world" ;
    4. memset(arr, 6, 5);
    5. return 0;
    6. }

    身为初学者,自知有很多不足,希望得到大佬们指点和改善!!!

     

  • 相关阅读:
    一套完善的设备管理系统能给企业带来什么?
    微信多开助手分享,想开几个就几个
    练习41,[SCOI2010] 序列操作【线段树】
    Zookeeper【Curator客户端Java版】从0到1——万字学习笔记
    MATLAB基本语法
    数据库设计原则
    AIGC AI绘画 Midjourney 的详细使用手册
    【Unity Shader】自定义变体使用
    麒麟操作系统设置QT程序开机自启动有效方法
    uniapp 打包小程序体积优化思路、优先排查优化项参考
  • 原文地址:https://blog.csdn.net/m0_67367079/article/details/133081848