c++中包含string成员的结构体拷贝导致的double free问题

简介: c++中包含string成员的结构体拷贝导致的double free问题

最近调试代码遇到一个的问题,提示double free,但是找了好久也没有找到释放两次的地方,后来调试发现,是由于使用了一个包含string成员的结构体,这个结构体使用memcpy拷贝导致的问题;

 

代码如下:

 
#include <stdio.h>
#include <map>
#include <string>
#include <stdlib.h>
#include <memory>
#include <iostream>
#include <string.h>
 
using namespace std;
 
typedef struct TestPtrInfo
{
    string name;
    int data;
}TestPtrInfoSt;
 
class testPtr
{
public:
    testPtr(int i,TestPtrInfoSt* info):m_id(i)
    {
        memcpy(&m_info, info, sizeof(TestPtrInfoSt));  //导致出问题的这一行
        cout<<m_id<<" name: "<<m_info.name<<" testPtr start..."<<endl;
    }
 
    ~testPtr()
    {
        cout<<m_id<<" name: "<<m_info.name<<" testPtr end..."<<endl;
    }
private:
    int m_id;
    TestPtrInfoSt m_info;
};
 
 
int main()
{
 
    //map<string, shared_ptr<testPtr> >testMap;
    map<string, testPtr* >testMap;
  
  TestPtrInfoSt info;
  for (int i = 0; i < 10 ; i++ )
  {
    info.name = "no-"+to_string(i);
    testMap[to_string(i)] = new testPtr(i, &info);    
  }
    
 
    auto iterPtr = testMap.find("1");
  delete iterPtr->second;
    testMap.erase(iterPtr);
    cout <<"1, reslease "<<endl;
 
  return 0;
}


这个不是正式代码,用测试代码复现的问题;

编译没问题,运行时会挂掉:


如上图,会报段错误,用gdb调试,打印堆栈信息如下:


从上面的堆栈信息中,看到这么一行:

#2  0xb7f4a985 in std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >::~basic_string() ()
   from /usr/lib/i386-linux-gnu/libstdc++.so.6


这里是string类的析构函数,在这里想到可能是memcpy拷贝结构体导致的,修改后测试不会再出现问题。


关于包含类的结构体,一般不能使用memcpy拷贝,会出问题。


参见:


memcpy复制字符串的注意事项/memcpy不能用来拷贝类类型 - hchacha - 博客园

https://www.cnblogs.com/hchacha/p/7615631.html


目录
相关文章
|
3月前
|
安全 编译器 程序员
【C++篇】C++类与对象深度解析(六):全面剖析拷贝省略、RVO、NRVO优化策略
【C++篇】C++类与对象深度解析(六):全面剖析拷贝省略、RVO、NRVO优化策略
64 2
|
3月前
|
存储 编译器 C++
【C++】深入探索类和对象:初始化列表及其static成员与友元(一)
【C++】深入探索类和对象:初始化列表及其static成员与友元
|
2月前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
123 5
|
3月前
|
安全 程序员 C++
双重释放(Double Free)
【10月更文挑战第12天】
159 2
|
3月前
|
数据可视化 Java
让星星月亮告诉你,通过反射创建类的实例对象,并通过Unsafe theUnsafe来修改实例对象的私有的String类型的成员属性的值
本文介绍了如何使用 Unsafe 类通过反射机制修改对象的私有属性值。主要包括: 1. 获取 Unsafe 的 theUnsafe 属性:通过反射获取 Unsafe类的私有静态属性theUnsafe,并放开其访问权限,以便后续操作 2. 利用反射创建 User 类的实例对象:通过反射创建User类的实例对象,并定义预期值 3. 利用反射获取实例对象的name属性并修改:通过反射获取 User类实例对象的私有属性name,使用 Unsafe`的compareAndSwapObject方法直接在内存地址上修改属性值 核心代码展示了详细的步骤和逻辑,确保了对私有属性的修改不受 JVM 访问权限的限制
75 4
|
3月前
|
编译器 C语言 C++
C++入门3——类与对象2-2(类的6个默认成员函数)
C++入门3——类与对象2-2(类的6个默认成员函数)
42 3
|
3月前
|
编译器 C语言 C++
详解C/C++动态内存函数(malloc、free、calloc、realloc)
详解C/C++动态内存函数(malloc、free、calloc、realloc)
563 1
|
3月前
|
存储 编译器 C++
C++入门3——类与对象2-1(类的6个默认成员函数)
C++入门3——类与对象2-1(类的6个默认成员函数)
60 1
|
3月前
|
存储 编译器 数据安全/隐私保护
【C++篇】C++类与对象深度解析(四):初始化列表、类型转换与static成员详解2
【C++篇】C++类与对象深度解析(四):初始化列表、类型转换与static成员详解
52 3
|
3月前
|
编译器 C++
【C++篇】C++类与对象深度解析(四):初始化列表、类型转换与static成员详解1
【C++篇】C++类与对象深度解析(四):初始化列表、类型转换与static成员详解
62 3