字符串函数内存函数(从零到一)【C语言】

长度不受限制的字符串函数:strcpy,strcat,strcmp

长度受限制的字符串函数:strncpy,strncat,strncmp

 strlen

 strlen函数是库函数中我们最常使用的函数,它可以求出字符串的长度(不包含‘\0’)

使用方法

通过前面对strlen函数的模拟实现我们知道strlen计算长度时,是遇到'\0'才停止的,因此如果一个字符串中没有'\0',程序就会输出一个随机值,如下面代码:

 因此我们在使用该函数时,必须避免这种情况。

注意:库函数中的strlen函数的类型是size_t(无符号整形)

模拟实现该函数的功能

这个函数我们实现的次数已经够多了,所以不再过多赘述,在此提供几种方法:

1.指针- 指针          2.计数器              3.递归

strcpy 

 strcpy函数用于将源字符串复制到目标字符串中,包括字符串的结束符'\0'。

使用方法

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

在使用strcpy函数时也要避免缺少'\0'的情况,此外,目标字符串必须具有足够的空间来存储源字符串的内容。

 模拟实现该函数的功能

#include<stdio.h>
#include<string.h>
void get_strcpy(char* des,const char* sou)//此处加const的原因是防止原字符串的内容被修改
{assert(sou != NULL);while (*des++ = *sou++){;	}	
}
int main()
{char arr1[10] = " xxxxxxx ";char arr2[10] = "bit";get_strcpy(arr1, arr2);printf("%s\n", arr1);return 0;
}

strcat 

strcat是一个字符串追加函数,它可以把一个字符串追加给另一个字符串后面。

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

使用方法

char *strcpy( char *strDestination, const char *strSource );

 

 模拟实现该函数的功能

#include<stdio.h>
char* my_strcat(char* dest, char* sou)
{char* ori = dest;//找到源字符串中的\0while (*dest){dest++;}//开始追加while (*dest++ = *sou++){;}return ori;
}
int main()
{char arr1[30] = "hello ";char arr2[] = "world";printf("%s\n",my_strcat(arr1, arr2));return 0;
}

注意事项

字符串不能自己给自己追加,通过模拟实现该函数我们发现,追加的标志是'\0',在追加开始时,'\0'就被自己的第一个元素覆盖,原先在准备追加最后一位'\0'的位置已经变成了一个正常的元素,所以循环将一直继续,不会终止。

strstr

该函数的功能是在一个字符串中查找另一个函数,如果查找到,则返回起始位置的地址,如果找不到,则返回一个空指针。

使用方法

char *strstr( const char *string, const char *strCharSet );

第一个参数是源字符串,第二个参数是要查找的字符串。 

模拟实现该函数的功能

#include<stdio.h>
#include<assert.h>
char* my_strstr(const char* str1, const char* str2)
{assert(str1 && str2);const char* s1 = NULL;const char* s2 = NULL;const char* tmp = str1;if (*str2 == '\0'){return (char*)str1;}while (*tmp){s1 = tmp;s2 = str2;while (*s1 && *s2 && (*s1 == *s2)){s1++;s2++;}if (*s2 == '\0'){return (char*)tmp;}tmp++;}return NULL;
}
int main()
{char arr1[30] = "aabbccd";char arr2[] = "bcc";char* pa = my_strstr(arr1, arr2);printf("%s\n", pa);return 0;
}

 memcpy

memcpy函数是一个内存操作函数,用于在内存中进行数据的复制。它将指定长度的数据从源地址复制到目标地址,不考虑源地址和目标地址是否重叠。当源地址和目标地址有重叠时,使用memcpy函数可能导致不可预测的结果。

该函数出现的原因:字符串的拷贝可以使用strcpy函数来进行,但是通过模拟实现strcpy的功能时,我们可以发现,strcpy拷贝结束的标志是‘\0’,但是如果拷贝的是一个整形数组,该函数就不再适用了,因此出现了这种更为通用的函数memcpy(内存拷贝函数)

该函数基本的使用方法:void *memcpy( void *dest, const void *src, size_t count );

使用方法

在msdn上可以查询出该函数的基本使用方式,前两个参数都是void类型的指针,意味着我们可以传入任意类型的指针参数,后面的size_t是无符号的整形,单位是字节,意味着操作者可以传入自己想要拷贝的字节。

模拟实现该函数的功能

当你对一个指针进行加1操作时,实际上会将指针的值增加一个与指针所指向的数据类型大小相等的偏移量。这个偏移量是以字节为单位计算的。

举个例子,如果一个指针指向整型数据(int),那么对该指针进行加1操作后,指针将向后移动4个字节(因为int类型通常占用4个字节)。类似地,如果指针指向字符型数据(char),那么对该指针进行加1操作后,指针将向后移动1个字节。

#include<stdio.h>
#include<assert.h>
void* my_memcpy(void* dest, const void* sou, size_t k)//k的单位是字节
{assert(dest && sou);void* orignal = dest;while (k--){*(char*)dest = *(char*)sou;dest = (char*)dest + 1;sou = (char*)sou + 1;}return orignal;
}
int main()
{int arr1[10] = { 1,2,3,4,5,6,7,8,9,10 };int arr2[10] = { 0 };my_memcpy(arr2, arr1, 20);int i = 0;for (i = 0; i < 10; i++){printf("%d ", arr2[i]);}return 0;
}

 标准功能的memcpy函数其实不能实现内存重叠的拷贝,比如说对于数组int arr[10] = {1,2,3,4,5,6,7,8,9,10 };我们如果使用memcpy(arr+2,arr,20),结果应该是1,2,1,2,1,2,7,8,9,10.因为存在内存重叠。但是VS上却能够正常编译,因为VS把这个函数的功能做的比较彻底,算是PLUS版本的memcpy函数,超出了本身固有的功能。对于内存重叠的拷贝我们其实还有一个特定的函数memmove,下面展开介绍。

memmove

 memmove 函数是一个内存操作函数,用于在内存中移动一段数据。它可以处理源地址和目标地址有重叠的情况,保证数据的正确复制。

使用方法

其使用方法与memcpy相似。 

模拟实现该函数的功能 

通过分析,我们发现,为保证字符串拷贝不受内存重叠的影响,当dest的指针小于sou的指针的时候要从前向后拷贝,当dest的指针大于等于sou的指针的时候要从后向前拷贝,当dest的指针大于sou+宽度时,不管是从后向前还是从前到后拷贝都可以。

总结:当dest的指针小于sou的指针的时候从前向后拷贝,dest的指针大于等于sou的指针的时候从后向前拷贝。

#include<stdio.h>
#include<string.h>
char* my_memmove(void* dest, const void* sou, size_t k)
{char* original = dest;if (dest < sou){while (k--){*(char*)dest = *(char*)sou;dest = (char*)dest + 1;sou = (char*)sou + 1;}}else{while (k--){*((char*)dest + k) = *((char*)sou + k);}}
}
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };my_memmove(arr + 2, arr, 20);int i = 0;for (i = 0; i < 10; i++){printf("%d ", arr[i]);}return 0;
}

memcmp

memcmp与strcmp的函数设计相似, 比较特定的字节后,如果值相同,返回值为零;如果不同,返回值是一个负数;如果相同,返回一个正数。

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

ptr1和ptr2分别是要比较的内存区域的起始地址,num是要比较的字节数。

使用方法

#include <stdio.h>
#include <string.h>int main()
{char str1[] = "Hello";char str2[] = "World";int result = memcmp(str1, str2, sizeof(str1));if (result == 0){printf("字符串相等\n");} else if (result < 0) {printf("str1小于str2\n");} else {printf("str1大于str2\n");}return 0;
}

memset

memset是库函数中的一个函数,用于将一段内存块的值设置为指定的值 ,

void *memset(void *ptr, int value, size_t num);

使用方法 

需要注意的是,memset函数只能设置每个字节的值,因此在设置非字符类型的数组时需要小心。另外,memset函数只能设置为整数值,不能设置为其他类型的值。 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/288724.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

c语言力扣题目:消失的数字(有关时间复杂度O(N²)O(N))以及对异或操作符的更深入的理解(如何用人脑的十进制去考量二进制)

目录 Way One :暴力求解,时间复杂度为 O(N) 代码1 Way Two : 时间复杂度限制到 O(N) 代码及其详解 如题 Way One :暴力求解,时间复杂度为 O(N) 大体思路:比如这里我们需要处理的整型数组是"3,0,1",我们可以用冒泡排序或者 qsort函数将他从大到小进行排序成"…

【数组Array】力扣-1094 拼车

目录 题目描述 解题过程 题目描述 车上最初有 capacity 个空座位。车 只能 向一个方向行驶&#xff08;也就是说&#xff0c;不允许掉头或改变方向&#xff09; 给定整数 capacity 和一个数组 trips , trip[i] [numPassengersi, fromi, toi] 表示第 i 次旅行有 numPassen…

理解pom.xml中的parent标签

✅作者简介&#xff1a;大家好&#xff0c;我是Leo&#xff0c;热爱Java后端开发者&#xff0c;一个想要与大家共同进步的男人&#x1f609;&#x1f609; &#x1f34e;个人主页&#xff1a;Leo的博客 &#x1f49e;当前专栏&#xff1a; 循序渐进学SpringBoot ✨特色专栏&…

06:2440----异常与中断

目录 一:概念的引入 1:ARM系统 2:CPU处理中断的过程 3:ARM对异常&#xff08;中断&#xff09;处理的过程 4:软/硬件工作分配 二:CPU模式状态与寄存器 1:ARM CPU的七种工作模式 2:state 3:寄存器 A:寄存器情况 B:CPSR C:状态位和保留位 D:协同工作 E:异常向量表 …

Text2SQL学习整理(四)将预训练语言模型引入WikiSQL任务

导语 上篇博客&#xff1a;Text2SQL学习整理&#xff08;三&#xff09;&#xff1a;SQLNet与TypeSQL模型简要介绍了WikiSQL数据集提出后两个早期的baseline&#xff0c;那时候像BERT之类的预训练语言模型还未在各种NLP任务中广泛应用&#xff0c;因而作者基本都是使用Bi-LSTM…

广东建筑模板价格一览表

在建筑行业&#xff0c;合适的建筑模板是确保工程顺利进行的关键材料之一。在选择建筑模板时&#xff0c;除了质量、材质等因素外&#xff0c;价格也是一个重要的考虑因素。本文将提供一个广东建筑模板的价格一览表&#xff0c;以供业内人士参考。需要注意的是&#xff0c;以下…

SpringSecurity入门

前言 Spring Security是一个用于在Java应用程序中提供身份验证和授权功能的强大框架。它构建在Spring框架之上&#xff0c;为开发人员提供了一套灵活且全面的安全性服务&#xff0c;本篇将为大家带来Spring Security的详细介绍及入门 一.安全框架 在学习了解Spring Security之…

【qt信号槽-5】信号槽相关注意事项记录

背景&#xff1a; 信号槽是qt很重要的概念&#xff0c;遇到问题帮助没少看。其中就有signals and slots这一章节&#xff0c;说得很到位。 概念琐碎&#xff0c;记录备忘。不对之处望指正。 【qt信号槽-1】槽函数重写问题&#xff0c;qt_metacall和qt_static_metacall-CSDN博…

【数据结构】并查集的简单实现,合并,查找(C++)

文章目录 前言举例&#xff1a; 一、1.构造函数2.查找元素属于哪个集合FindRoot3.将两个集合归并成一个集合Union4.查找集合数量SetCount 二、源码 前言 需要将n个不同的元素划分成一些不相交的集合。开始时&#xff0c;每个元素自成一个单元素集合&#xff0c;然后按一定的规…

交叉验证之KFold和StratifiedKFold的使用(附案例实战)

&#x1f935;‍♂️ 个人主页&#xff1a;艾派森的个人主页 ✍&#x1f3fb;作者简介&#xff1a;Python学习者 &#x1f40b; 希望大家多多支持&#xff0c;我们一起进步&#xff01;&#x1f604; 如果文章对你有帮助的话&#xff0c; 欢迎评论 &#x1f4ac;点赞&#x1f4…

文件操作学习总结

磁盘上的⽂件是⽂件。 但是在程序设计中&#xff0c;我们⼀般谈的⽂件有两种&#xff1a; 程序⽂件、数据⽂件 &#xff08;从⽂件功能的⻆度来分类 的&#xff09;。 程序⽂件 &#xff1a; 程序⽂件包括源 程序⽂件&#xff08;后缀为.c&#xff09; , ⽬标⽂件&#xff0…

Opencv入门四(宽高为原始图像的一半)

代码如下&#xff1a; #include <opencv2/opencv.hpp> int main(int argc, char** argv) { cv::Mat img1, img2; cv::namedWindow("Example1", cv::WINDOW_AUTOSIZE); cv::namedWindow("Example2", cv::WINDOW_AUTOSIZE); img1 cv…