这可能跟你的那几个strncpy的使用方式有关。我查了一下这个函数的手册,大致是这么说。这个函数在源比size大时,copy size个字符,不包括null。在源小于size时,向目标填充足够的null,直至size指定的大小,这是标准C规定的。
这意味着你的调用点会造成多个缓冲区越界,因为它们都被追加了很多0(我想你的几个参数应该都不到100大小吧?)
试着用正确的方式改写那部分再试试
任何时候都应该检查目标缓冲区是否有足够的空间容纳size个字节,当用memcpy或者strncpy这类函数时。
或空间是否不小于源缓冲区,当用strcpy这类自动检查null结尾的函数时,还要算上copy会自动包含的null结尾。
p2[5] 这样直接在栈中分配了5个字节。
char p2 这样只是申请了一个指针,它应该指向一个可访问的内存地址。
刚分配出来很可能是NULL (0)也就是说 p2 的值是 0
那么你的memcpy(p2,p1,3);就是往 0 这个内存地址复制3个字节,会出现错误
0xXXXXXXXX 指令引用的地址 0x00000000不能为write……
正确的方法是给指针使用堆内存。
char p2 = malloc(5);
需要注意的是:使用堆内存需要回收,它不像栈内存在一个环境中自动生成销毁,堆内存需要手动回收。其指令是
free(p2);
取决于编译器memcpy实现的方式
一般有三种,从头复制 从尾复制 或者判断复制(等同于memmove)
仿照你的 我写了一个测试程序
char message2[60] = "abcdefghijklmnopqrstuvwxyz";char tmp[60];
int main()
{
strcpy(tmp, message2);
printf("tmp= \"%s\"\n", tmp);
memcpy(tmp+4, tmp + 16, 10);
printf("memcpy(tmp+4, tmp + 16, 10);> tmp = \"%s\"\n", tmp);
strcpy(tmp, message2);
memcpy(tmp+6, tmp +4, 10);
printf("memcpy(tmp+6, tmp +4, 10);> tmp = \"%s\"\n", tmp);
strcpy(tmp, message2);
memcpy(tmp+4, tmp +6, 10);
printf("memcpy(tmp+4, tmp +6, 10);> tmp = \"%s\"\n", tmp);
}
测试结果如下
tmp= "abcdefghijklmnopqrstuvwxyz"
memcpy(tmp+4, tmp + 16, 10);> tmp = "abcdqrstuvwxyzopqrstuvwxyz"
memcpy(tmp+6, tmp +4, 10);> tmp = "abcdefefefijijmnqrstuvwxyz"
memcpy(tmp+4, tmp +6, 10);> tmp = "abcdghijklmnopopqrstuvwxyz"
可以看到 在memcpy(tmp+6, tmp +4, 10);时出现了覆盖 efefef
说明我的编译器(gcc)在做memcpy的时候使用的是头复制 即从起始地址开始复制
如果这个不出错 而memcpy(tmp+4, tmp +6, 10);出错 那么就是尾复制
都不出错的话 那么就是判断之后智能复制 相当于memmove
memcpy函数用法
memcpy指的是c和c++使用的内存拷贝函数,memcpy函数的功能是从源内存地址的起始位置开始拷贝若干个字节到目标内存地址中。
函数原型
void memcpy(void dest, const void src, size_t n);
折叠编辑本段功能
从源src所指的内存地址的起始位置开始拷贝n个字节到目标dest所指的内存地址的起始位置中
折叠编辑本段所需头文件
C语言:#include<stringh>
C++:#include<cstring>
折叠编辑本段返回值
函数返回指向dest的指针。
折叠编辑本段说明
1source和destin所指的内存区域可能重叠,但是如果source和destin所指的内存区域重叠,那么这个函数并不能够确保source所在重叠区域在拷贝之前不被覆盖。而使用memmove可以用来处理重叠区域。函数返回指向destin的指针
2如果目标数组destin本身已有数据,执行memcpy()后,将覆盖原有数据(最多覆盖n)。如果要追加数据,则每次执行memcpy后,要将目标数组地址增加到你要追加数据的地址。
注意:source和destin都不一定是数组,任意的可读写的空间均可。
折叠编辑本段函数实现
Windows中
coreutils中
Linux中:
折叠编辑本段程序实例
程序例example1
作用:将s中的字符串复制到字符数组d中。
输出结果:Golden Global View
example2
作用:将s中第13个字符开始的4个连续字符复制到d中。(从0开始)
输出结果: View
example3
作用:复制后覆盖原有部分数据
输出结果:
destination before memcpy:abcdefghijlkmnopqrstuvwxyz0123as6
destination after memcpy: as6
strcpy和memcpy主要有以下3方面的区别。
1、复制的内容不同。strcpy只能复制字符串,而memcpy可以复制任意内容,例如字符数组、整型、结构体、类等。
2、复制的方法不同。strcpy不需要指定长度,它遇到被复制字符的串结束符"\0"才结束,所以容易溢出。memcpy则是根据其第3个参数决定复制的长度。
3、用途不同。通常在复制字符串时用strcpy,而需要复制其他类型数据时则一般用memcpy
是覆盖
函数名: memcpy
功 能: 从源source中拷贝n个字节到目标destin中
用 法: void memcpy(void destin, void source, unsigned n);
程序例:
#include <stdioh>
#include <stringh>
int main(void)
{
char src[] = "";
char dest[] = "abcdefghijlkmnopqrstuvwxyz0123456709";
char ptr;
printf("destination before memcpy: %s\n", dest);
ptr = memcpy(dest, src, strlen(src));
if (ptr)
printf("destination after memcpy: %s\n", dest);
else
printf("memcpy failed\n");
return 0;
}
memcpy_s包含在stringh头文件中
另外memcpy_s函数,在C11开始才加入C标准,所以编译器要支持C11才能正常使用
以上就是关于C语言free的时候程序出错全部的内容,包括:C语言free的时候程序出错、memcpy的使用问题、关于memcpy()函数等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)