模拟实现库函数strcpy,对strcpy的进一步理解(深刻理解重叠问题,防止内存与源重叠)

简介: 模拟实现库函数strcpy,对strcpy的进一步理解(深刻理解重叠问题,防止内存与源重叠)

一.对库函数strcpy的了解



通过在MSDN或者cplusplus网站上检索strcpy

dc24ed1326c2474d9a75139053cbb27c.png

通过对strcpy的检索,可以初步了解到strcpy以下信息:


1.将源头得字符串复制到目标数组中

2.传入的参数只能是指针

3.连同 \0 一起会被复制到目标数组中

4.返回类型为char* 类型,即目标数组的地址

5.目标素组的内存大小应该要能放的小源头字符串并且不溢出


二.模拟实现库函数strcpy



#include<assert.h>
#include<stdio.h>
//将源头数据拷贝到目标数组,源头数据不可修改,用const限制
char* my_strcpy(char* dest, const char* soc)
{
  assert(dest && soc);//断言dest和soc不是空指针
  //记录目标数据得起始位置
  char* p = dest;
  //将包括\0在内soc里得内容拷贝到dest里
  while (*dest++ = *soc++) 
  {
    ;
  }
  return p;  //返回目标地址
}
int main()
{
  char arr1[50] = { 0 };
  char arr2[] = "abcdef";
  my_strcpy(arr2+1, arr2);
  printf("%s",arr2);
  return 0;
}


三.strcpy的一些特殊情况



1.第一种情况

源头数据与目标数据内存重叠时(目标数据覆盖在源头数据里面

e3f2211d427d4715a2bed54190f9123f.png



#include<assert.h>
#include<stdio.h>
char* my_strcpy(char* dest, const char* soc)
{
  assert(dest && soc);
  //记录目标数据得起始位置
  char* p = dest;
  //将包括\0在内soc里得内容拷贝到dest里
  while (*dest++ = *soc++) 
  {
    ;
  }
  return p;  //返回目标地址
}
int main()
{
  char arr1[50] = { 0 };
  char arr2[10] = "abcd";
  my_strcpy(arr2+2, arr2);
  printf("%s",arr2);
  return 0;
}


运行结果如下:

bc2c2202d7dd4886bbe501aed4d553e6.png


通过分析,对于条件while(*dest++=*soc++),soc找不到 \0 复制给dest,导致条件无法停止下来。

不难发现,当目标内存和源头内存重叠的时候,程序无法进行,会直接崩溃掉


2.第二种情况


.目标数据与源头数据重叠时(源头数据覆盖在目标数据里面

2db5e77662c44fa7a192c7b106122932.png


#include<assert.h>
#include<stdio.h>
char* my_strcpy(char* dest, const char* soc)
{
  assert(dest && soc);
  //记录目标数据得起始位置
  char* p = dest;
  //将包括\0在内soc里得内容拷贝到dest里
  while (*dest++ = *soc++) 
  {
    ;
  }
  return p;  //返回目标地址
}
int main()
{
  char arr1[50] = { 0 };
  char arr2[10] = "abcd";
  my_strcpy(arr2, arr2+2);
  printf("%s",arr2);
  return 0;
}


通过分析,预期结果为: c d

运行结果为:

82e1d3c25b3b4d0085a6707fcf874f85.png


不难发现,虽然目标数据与源头数据有重合,但是当源头数据覆盖在目标数据里面时,
\0可以顺利被拷贝进去,可以正常运行并且实现


四.总结



1 .对于strcpy得模拟实现中,通过上述分析,只要源头数据能拷贝到 \0,即可。若要完成第一种情况,可以参考使用memmove这个库函数。


2.对于cpluscplus对于strcpy得定义中所说得内存中不应与源重叠为第一种情况,关于内存不应与源重叠得具体理解,可以参考另一篇文章–模拟实现库函数memmove


相关文章
|
4天前
|
Java 程序员 Linux
探索C语言宝库:从基础到进阶的干货知识(类型变量+条件循环+函数模块+指针+内存+文件)
探索C语言宝库:从基础到进阶的干货知识(类型变量+条件循环+函数模块+指针+内存+文件)
11 0
|
4天前
|
程序员 C语言
C语言内存管理:malloc、calloc、realloc与free函数详解
C语言内存管理:malloc、calloc、realloc与free函数详解
6 0
|
7天前
|
C语言
【C语言】:4大内存函数
【C语言】:4大内存函数
11 2
|
7天前
|
C语言
【C语言】:动态内存管理函数malloc,calloc,realloc和free的介绍的介绍
【C语言】:动态内存管理函数malloc,calloc,realloc和free的介绍的介绍
18 0
|
8天前
|
C语言
C语言内存函数
C语言内存函数
8 0
|
10天前
|
C语言 C++
C语言----C语言内存函数
C语言----C语言内存函数
10 0
|
11天前
10分钟让你学会内存函数:memcpy,memmove,memset,memcmp的用法及模拟实现。
10分钟让你学会内存函数:memcpy,memmove,memset,memcmp的用法及模拟实现。
12 2
|
23小时前
|
存储
|
6天前
|
存储 C语言
【C语言进阶篇】整数在内存的存储——原码、反码、补码
【C语言进阶篇】整数在内存的存储——原码、反码、补码
|
8天前
|
存储 Java C++
Java虚拟机(JVM)管理内存划分为多个区域:程序计数器记录线程执行位置;虚拟机栈存储线程私有数据
Java虚拟机(JVM)管理内存划分为多个区域:程序计数器记录线程执行位置;虚拟机栈存储线程私有数据,如局部变量和操作数;本地方法栈支持native方法;堆存放所有线程的对象实例,由垃圾回收管理;方法区(在Java 8后变为元空间)存储类信息和常量;运行时常量池是方法区一部分,保存符号引用和常量;直接内存非JVM规范定义,手动管理,通过Buffer类使用。Java 8后,永久代被元空间取代,G1成为默认GC。
19 2