C语言编写字符串拷贝(strcpy)函数详解以及assert函数

  • Post author:
  • Post category:其他



目录


一.strcpy函数


原型声明


功能


说明


代码及运行结果


二.自己编写strcpy函数


代码一及运行结果


代码二(改进)及运行结果


代码三(进一步改进)及assert函数


在这里解释一下什么是asser函数(断言函数)


举个简单例子


代码四(最终改进)


一.strcpy函数

原型声明

char *strcpy(char* dest, const char *src);

功能

把从src地址开始且含有NULL结束符的字符串复制到以dest开始的地址空间,也就是把含有转义字符’\0’即空字符作为结束符,把这之前的内容复制到dest中。

说明

src和dest所指内存区域不可以重叠目

dest必须有足够的空间来容纳src的字符串

。如果目标数组dest 不够大,而源字符串的长度又太长,可能会造成缓冲溢出的情况。

头文件:#include <string.h>和#include <stdio.h>

代码及运行结果

#include<stdio.h>
#include<string.h>
int main()
{
    char arr1[]="############";
    char arr2[]="hello world!";
    printf("arr1: %s\n",arr1);
    strcpy(arr1,arr2);
    printf("把arr2中的内容赋值到arr1中去,arr1:%s",arr1);
    return 0;
}

结果是:

二.自己编写strcpy函数

代码一及运行结果

写一个while循环,只要src没有达到‘\0’,就一直把src的内容赋值给dest,然后不断++操作就可以了

#include<stdio.h>
#include<string.h>
void my_strcpy(char* dest,char* src)
{
    while(*src!='\0')
    {
        *dest=*src;
        dest++;
        src++;
    }
    *dest=*src;//把'\0'也传过去
}
int main()
{
    char arr1[]="############";
    char arr2[]="hello world!";
    my_strcpy(arr1,arr2);
    printf("arr1: %s\n",arr1);
    return 0;
}

运行结果:

代码二(改进)及运行结果

还可以对代码一中的循环进行改进,使代码更简洁

#include<stdio.h>
#include<string.h>
void my_strcpy(char* dest,char* src)
{
    while(*dest++=*src++)//在这里把‘\0’也传过去了
    {
        ;
    }
}
int main()
{
    char arr1[]="#################";//这里有17个#
    char arr2[]="hello world!";
    my_strcpy(arr1,arr2);
    printf("arr1: %s\n",arr1);
    printf("%d %d\n",sizeof(arr1),strlen(arr1));
    return 0;
}

运行结果:

这里为了让大家更能够理解发生了什么,我还分别输出了:sizeof(arr1)和strlen(arr1)的结果,我们发现,sizeof(arr1)的结果是18,strlen(arr1)的结果是12,所以说我们就知道,

arr1数组经过函数变化后它的结果是这样的“hello world!\0######”,

这样才会输出sizeof(arr1)的结果是18,strlen(arr1)的结果是12,因为strlen函数会输出以’\0’为结束字符前面的字符个数。

代码三(进一步改进)及assert函数

我们希望dest或者src如果他们输入不符合规则能够停止运行

#include<stdio.h>
#include<string.h>
#include<assert.h>
void my_strcpy(char* dest,char* src)
{

    assert(dest !=NULL);//断言
    assert(src !=NULL);//断言
    while(*dest++ = *src++)//在这里把‘\0’也传过去了
    {
        ;
    }
}
int main()
{
    char arr1[]="#################";//这里有17个#
    char arr2[]="hello world!";
    my_strcpy(arr1,arr2);
    printf("arr1: %s\n",arr1);
    printf("%d %d\n",sizeof(arr1),strlen(arr1));
    return 0;
}

在这里解释一下什么是asser函数(断言函数)

断言函数,用于在调试过程中捕捉程序的错误。在编程中是指对某种假设条件进行检测,

如果条件成立就不进行任何操作,如果条件不成立就捕捉到这种错误,并打印出错误信息,终止程序执行。但是需注意:频繁的调用assert会极大的影响程序的性能,增加额外的开销。

举个简单例子

我们用被除数不能为0写一个代码看看

#include<stdio.h>
#include<assert.h>
int main()
{
    int a,b;
    scanf("%d %d",&a,&b);
    assert(b!=0);
    int c=a/b;
    printf("c=%d",c);
}

assert(b!=0),就是在这里规定b不能为0,如果为0就报错终止执行,我们看看结果


代码四(最终改进)

我们有时候会把dest和src两个的顺序搞混,我们注意到strcpy函数原型

src前面自带一个const,使其变成常变量,这样就可以保证就算dest和src两者的顺序写反了,也不会出现将dest中的内容赋值到src中去

#include<stdio.h>
#include<string.h>
#include<assert.h>
void my_strcpy(char* dest,const char* src)
{

    assert(dest !=NULL);//断言
    assert(src !=NULL);//断言
    while(*dest++ = *src++)//在这里把‘\0’也传过去了
    {
        ;
    }
}
int main()
{
    char arr1[]="#################";//这里有17个#
    char arr2[]="hello world!";
    my_strcpy(arr1,arr2);
    printf("arr1: %s\n",arr1);
    printf("%d %d\n",sizeof(arr1),strlen(arr1));
    return 0;
}

用const就可以避免拷贝的方向、去处错误,保证了src不变,所以只能将src的内容拷贝到dest,而不能把dest里面的内容拷贝到src



版权声明:本文为weixin_63318044原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。