欢迎来到天天文库
浏览记录
ID:1670504
大小:52.50 KB
页数:4页
时间:2017-11-13
《c语言中内存重叠问题》由会员上传分享,免费在线阅读,更多相关内容在学术论文-天天文库。
1、内存重叠:拷贝的目的地址在源地址范围内。所谓内存重叠就是拷贝的目的地址和源地址有重叠。在函数strcpy和函数memcpy都没有对内存重叠做处理的,使用这两个函数的时候只有程序员自己保证源地址和目标地址不重叠,或者使用memmove函数进行内存拷贝。memmove函数对内存重叠做了处理。现在来看函数strcpy原型:externchar*strcpy(char*dest,char*source);功能:把source所指由NULL结束的字符串复制到dest所指的数组中。说明:source和dest所指内存区域不可以重叠且dest必须有足够的空间来容纳source的字符串。返回指
2、向dest的指针。重叠从两方面考虑:(1).dest数据覆盖了source;如:dest(8byte)地址:1000source(8byte)地址:1002(2).dest所指的区域本来就是source的一部分;如:dest(8byte)地址:1000source(8byte)地址:0998例如:针对第一种交叉情况情况,dstsrc,memcpy和memmove的结果是一样的。请看下面的例子讲解:strings="helloworld";memmove(&s[0],&s[5],10);举个内存重叠环境的例子:intmain(){char*p=NUL
3、L;p=(char*)malloc(100);memcpy(p,"123456789",strlen("123456789"));//会等到错误的结果,有一个长度参数,只能拷贝cnt个//字节就结束了printf("beforep=%s",p);strcpy(p+1,p);//注意:这里重叠了,而strcpy是根据判断原串中的' 'printf("afterp=%s",p);free(p);}1.下面来看strcpy()原型写法:字符串拷贝. char*strcpy(char*strDest,constchar*strSrc){assert((strDest!=NUL
4、L)&&(strSrc!=NULL));char*address=strDest; while((*strDest++=*strSrc++)·1!='/0') NULL; returnaddress; }2.下面来看下memcpy函数的原型写法:内存拷贝void*memcpy(void*dest,constvoid*source,size_tcount){assert((NULL!=dest)&&(NULL!=source));char*tmp_dest=(char*)dest;char*tmp_source=(char*)source;while(count--)//不对是否
5、存在重叠区域进行判断*tmp_dest++=*tmp_source++;returndest;}3.下面来看下memmove函数的原型写法:void*memmove(void*dest,constvoid*source,size_tcount){assert((NULL!=dest)&&(NULL!=source));char*tmp_source,*tmp_dest;tmp_source=(char*)source;tmp_dest=(char*)dest;if((dest+count6、7、(source+count)8、e(count--)*tmp_dest++=*tmp_source++;}else{//如果有重叠(反向拷贝)tmp_source+=count-1;tmp_dest+=count-1;while(count--)*--tmp_dest=*--tmp;}returndest;}深入分析:void*memcpy(void*dst,constvoid*src,size_tcount):void*memmove(void*dst,constvoid*src,size_tcount);先看一个测试:#include#includeintmain()9、{inta[10];for(inti=0;i<10;i++)a[i]=i;memcpy(&a[4],a,sizeof(int)*6);//结果为:123012301//memcpy(&a[4],a,sizeof(int)*6);//结果为:123012301(vc下和下面一个相同)//MemMove(&a[4],a,sizeof(int)*6);//结果为:123012345//memmove(&a[4],a,sizeof(int)*6);//结果为:123012345//MemMove(a
6、
7、(source+count)8、e(count--)*tmp_dest++=*tmp_source++;}else{//如果有重叠(反向拷贝)tmp_source+=count-1;tmp_dest+=count-1;while(count--)*--tmp_dest=*--tmp;}returndest;}深入分析:void*memcpy(void*dst,constvoid*src,size_tcount):void*memmove(void*dst,constvoid*src,size_tcount);先看一个测试:#include#includeintmain()9、{inta[10];for(inti=0;i<10;i++)a[i]=i;memcpy(&a[4],a,sizeof(int)*6);//结果为:123012301//memcpy(&a[4],a,sizeof(int)*6);//结果为:123012301(vc下和下面一个相同)//MemMove(&a[4],a,sizeof(int)*6);//结果为:123012345//memmove(&a[4],a,sizeof(int)*6);//结果为:123012345//MemMove(a
8、e(count--)*tmp_dest++=*tmp_source++;}else{//如果有重叠(反向拷贝)tmp_source+=count-1;tmp_dest+=count-1;while(count--)*--tmp_dest=*--tmp;}returndest;}深入分析:void*memcpy(void*dst,constvoid*src,size_tcount):void*memmove(void*dst,constvoid*src,size_tcount);先看一个测试:#include#includeintmain()
9、{inta[10];for(inti=0;i<10;i++)a[i]=i;memcpy(&a[4],a,sizeof(int)*6);//结果为:123012301//memcpy(&a[4],a,sizeof(int)*6);//结果为:123012301(vc下和下面一个相同)//MemMove(&a[4],a,sizeof(int)*6);//结果为:123012345//memmove(&a[4],a,sizeof(int)*6);//结果为:123012345//MemMove(a
此文档下载收益归作者所有