情况:
程序正常退出或函数正常结束或离开局部作用域时,
(xxx中)引发的异常,0xC0000005:写入位置 0xDDDDDDDD 时发生访问冲突.
先上错误结论:
使用memcpy时,被拷贝的对象里面存在动态内存.
比如:vector对象大小无法确定,memcpy不管这事直接拷贝sizeof大小的内存,导致vector的内存结构破坏,vector析构时就出错了
Update:
后面发现vector内存结构破坏这话说的不太靠谱,不太直观. 其实memcpy是可以拷贝vector对象的,但是使用时会发现代码依然会报错. 对,我想知道的是为什么拷贝后析构时会报错!!! 这才是我想说的.
结论:
使用memcpy时会拷贝vector对象中指向具体对象的指针. 当复制的对象被销毁时原对象指向的具体对象已经被销毁,所以原对象再次准备销毁时会报错
1. 问题代码:
struct xxx {
int ID;
std::vector<string> name;
};
{
xxx tmp1;
tmp1.name.push_back("小明");
tmp1.name.push_back("张三");
tmp1.name.push_back("李四");
xxx tmp22;
memcpy(&tmp22, &tmp1, sizeof(tmp1));
std::cout << sizeof(xxx) << std::endl;
std::cout << sizeof(tmp1) << std::endl;
std::cout << sizeof(tmp22) << std::endl;
std::cout << sizeof(tmp1.name[0]) + sizeof(tmp1.name[1]) + sizeof(tmp1.name[2]) ;
}
//在代码里加上这段程序之后就会莫名的崩溃,说是解析对象出错.
如图:
这个例子只能说明,vector<std::string>对象的大小是16,而vector对象里面的数据并不储存在vector对象内.
2. 问题代码
void test_vector_memcpy2()
{
std::vector<std::string> A1;
A1.push_back("小明");
A1.push_back("校长");
A1.push_back("张三");
std::cout << "A1" << std::endl;
for (size_t i = 0; i<A1.size();++i)
{
std::cout << &A1[i] << std::endl;
}
{
std::vector<std::string> A2;
memcpy(&A2, &A1, sizeof(A1));
std::cout << "A2" << std::endl;
for (size_t i = 0; i < A2.size();++i)
{
std::cout << &A2[i] << std::endl;
}
//离开作用域后会析构A2内的值
std::vector<std::string> A3;
A3 = A1; //如果使用自带拷贝
std::cout << "A3" << std::endl;
for (size_t i = 0 ; i < A3.size();++i )
{
std::cout << &A3[i] << std::endl;
}
}
// 导致再次访问A1的这些值会报错
for (size_t i = 0;i < A1.size() ; ++i)
{
std::cout << A1[i] << std::endl;
}
}
结果如图:
再看如果使用 vector的" = “进行复制,能发现使用”="进行复制会把vector指向的对象内容进行深度拷贝(即有不同的存储地址).