免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
12下一页
最近访问板块 发新帖
查看: 5022 | 回复: 18
打印 上一主题 下一主题

[C] strcpy重叠拷贝问题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2015-07-11 14:58 |只看该作者 |倒序浏览
本帖最后由 maozone 于 2015-07-11 15:53 编辑

麻烦请教一下,这段代码为什么会出现下面运行结果:
  1. #include <stdio.h>
  2. int main()
  3. {
  4.         int iLen;
  5.         char str[1024];
  6.         char sub[128];
  7.         char *p1;
  8.         char *p2;
  9.        
  10.         strcpy(str, "1,2,3,4,5,6,7,9,10,11,12,13,14,15,16,17,18,19,20");
  11.         strcpy(sub, "12");
  12.         printf("sub=%s\nstr=%s\n", sub, str);
  13.         p1 = strstr(str, sub);
  14.         iLen = strlen(sub) + 1;
  15.         p2 = p1 + iLen;
  16.         strcpy(p1, p2);
  17.         printf("str=%s\n", str);
  18. }
复制代码
sub=12
str=1,2,3,4,5,6,7,9,10,11,12,13,14,15,16,17,18,19,20
str=1,2,3,4,5,6,7,9,10,11,13,14,15,17,17,18,19,20
拷贝后结果不对,16没有了,而17重复。
strcpy这里改成memmove,或者直接用指针逐个赋值就没问题。
但是一般来说p2>p1时用strcpy应该不会有问题阿!

论坛徽章:
0
2 [报告]
发表于 2015-07-11 15:03 |只看该作者
操作系统:redhat6.3 64bit
GCC信息:
Target: x86_64-redhat-linux
Configured with: ../configure --prefix=/usr --mandir=/usr/share/man --infodir=/usr/share/info --with-bugurl=web://bugzilla.redhat.com/bugzilla --enable-bootstrap --enable-shared --enable-threads=posix --enable-checking=release --with-system-zlib --enable-__cxa_atexit --disable-libunwind-exceptions --enable-gnu-unique-object --enable-languages=c,c++,objc,obj-c++,java,fortran,ada --enable-java-awt=gtk --disable-dssi --with-java-home=/usr/lib/jvm/java-1.5.0-gcj-1.5.0.0/jre --enable-libgcj-multifile --enable-java-maintainer-mode --with-ecj-jar=/usr/share/java/eclipse-ecj.jar --disable-libjava-multilib --with-ppl --with-cloog --with-tune=generic --with-arch_32=i686 --build=x86_64-redhat-linux
Thread model: posix
gcc version 4.4.6 20120305 (Red Hat 4.4.6-4) (GCC)

论坛徽章:
36
子鼠
日期:2013-08-28 22:23:29黄金圣斗士
日期:2015-12-01 11:37:51程序设计版块每日发帖之星
日期:2015-12-14 06:20:00CU十四周年纪念徽章
日期:2015-12-22 16:50:40IT运维版块每日发帖之星
日期:2016-01-25 06:20:0015-16赛季CBA联赛之深圳
日期:2016-01-27 10:31:172016猴年福章徽章
日期:2016-02-18 15:30:3415-16赛季CBA联赛之福建
日期:2016-04-07 11:25:2215-16赛季CBA联赛之青岛
日期:2016-04-29 18:02:5915-16赛季CBA联赛之北控
日期:2016-06-20 17:38:50技术图书徽章
日期:2016-07-19 13:54:03程序设计版块每日发帖之星
日期:2016-08-21 06:20:00
3 [报告]
发表于 2015-07-11 15:12 |只看该作者
重叠               

论坛徽章:
36
子鼠
日期:2013-08-28 22:23:29黄金圣斗士
日期:2015-12-01 11:37:51程序设计版块每日发帖之星
日期:2015-12-14 06:20:00CU十四周年纪念徽章
日期:2015-12-22 16:50:40IT运维版块每日发帖之星
日期:2016-01-25 06:20:0015-16赛季CBA联赛之深圳
日期:2016-01-27 10:31:172016猴年福章徽章
日期:2016-02-18 15:30:3415-16赛季CBA联赛之福建
日期:2016-04-07 11:25:2215-16赛季CBA联赛之青岛
日期:2016-04-29 18:02:5915-16赛季CBA联赛之北控
日期:2016-06-20 17:38:50技术图书徽章
日期:2016-07-19 13:54:03程序设计版块每日发帖之星
日期:2016-08-21 06:20:00
4 [报告]
发表于 2015-07-11 15:18 |只看该作者
wait,两个17?

论坛徽章:
0
5 [报告]
发表于 2015-07-11 15:48 |只看该作者
本帖最后由 maozone 于 2015-07-11 15:52 编辑

是的,拷贝后结果不对.
strcpy这里改成memmove,或者直接用指针逐个赋值就没问题。
但是一般来说p2>p1时用strcpy应该不会有问题的阿

论坛徽章:
44
15-16赛季CBA联赛之浙江
日期:2021-10-11 02:03:59程序设计版块每日发帖之星
日期:2016-07-02 06:20:0015-16赛季CBA联赛之新疆
日期:2016-04-25 10:55:452016科比退役纪念章
日期:2016-04-23 00:51:2315-16赛季CBA联赛之山东
日期:2016-04-17 12:00:2815-16赛季CBA联赛之福建
日期:2016-04-12 15:21:2915-16赛季CBA联赛之辽宁
日期:2016-03-24 21:38:2715-16赛季CBA联赛之福建
日期:2016-03-18 12:13:4015-16赛季CBA联赛之佛山
日期:2016-02-05 00:55:2015-16赛季CBA联赛之佛山
日期:2016-02-04 21:11:3615-16赛季CBA联赛之天津
日期:2016-11-02 00:33:1215-16赛季CBA联赛之浙江
日期:2017-01-13 01:31:49
6 [报告]
发表于 2015-07-11 20:24 |只看该作者
strcpy的两个参数如果重叠,是一个UB。

论坛徽章:
0
7 [报告]
发表于 2015-07-12 08:20 |只看该作者
如果dest大于src重叠出现错误很好理解,但是为什么src大于dest也会有问题?这个是让人疑惑的地方,能否帮忙解答一下。
按照strcpy的实现
  1. *dest++ = *src++
复制代码
应该不会出现问题啊,实际测试也确实没有问题。

论坛徽章:
7
CU大牛徽章
日期:2013-03-14 14:16:29CU大牛徽章
日期:2013-03-14 14:16:32CU大牛徽章
日期:2013-03-14 14:16:34CU大牛徽章
日期:2013-03-14 14:16:35IT运维版块每日发帖之星
日期:2015-07-13 23:09:32IT运维版块每日发帖之星
日期:2015-07-13 22:20:00数据库技术版块每日发帖之星
日期:2015-09-08 06:20:00
8 [报告]
发表于 2015-07-12 09:43 |只看该作者
你不觉的你这样用指针非常糟糕么?
char *总要指向一个已经分配好地址空间的char []变量,如果是c++ ,也可以new一个。
你光定义一个char *p;这个一个游荡的变量,没有分配内存,是非常危险的

论坛徽章:
0
9 [报告]
发表于 2015-07-12 10:04 |只看该作者
本来就是这个结果,内存重叠用memmove吧。

论坛徽章:
1
2015年辞旧岁徽章
日期:2015-03-03 16:54:15
10 [报告]
发表于 2015-07-12 11:10 |只看该作者
maozone 发表于 2015-07-12 08:20
如果dest大于src重叠出现错误很好理解,但是为什么src大于dest也会有问题?这个是让人疑惑的地方,能否帮忙 ...


这个,怎么说呢,strcpy的实现并不是简单的

dest++ = src++;

memmove更加复杂
有兴趣的话建议你去看看glibc的源代码
为每一个支持的cpu都有对应的版本

不要用C/C++的语言特性去理解C/C++库的实现
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP