如何编写析构函数以便能够编译代码并释放所有分配的内存?

时间:2013-04-07 19:19:35

标签: c++ memory-management destructor

我已经创建了自定义类,它的行为就像一个矩阵。我有一些基本的操作,一切似乎都运行良好......但是,我无法找到,我应该写什么到这个类的析构函数来释放所有分配的内存。你能建议我吗?

class CMatrix {
public:

class Proxy {
    friend class CMatrix;

    const CMatrix* cmat;
    CMatrix *matrix;
    size_t n;

    Proxy(const CMatrix& m, size_t i)
    : cmat(&m), matrix(), n(i) {
    }

    Proxy(CMatrix& m, size_t i)
    : cmat(&m), matrix(&m), n(i) {
    }

    public:

    const double& operator[](size_t j) const {
       return cmat->_arrayofarrays[n * cmat->y + j];
    }

    double& operator[](size_t j) {
        if (matrix) {
            return matrix->_arrayofarrays[n * cmat->y + j];
        } else return cmat->_arrayofarrays[cmat->y];
    }
    };

    const Proxy operator[](size_t i) const {
        return Proxy(*this, i);
    }

    Proxy operator[](size_t i) {
        return Proxy(*this, i);
    }

    CMatrix() {
    _arrayofarrays = NULL;
    x = 0;
    y = 0;
};

// constructor

CMatrix(size_t x, size_t y) : _arrayofarrays(), x(x), y(y) {
    _arrayofarrays = new double[ x * y ]();
}

// destructor

~CMatrix() {
    // ?!?!???!?!?!?!!!!?!?!? 
    // #$#%@^$!!!!@$#%!!
}

// copy constructor

CMatrix(const CMatrix& other) : _arrayofarrays(), x(other.x), y(other.y) {
    delete [] _arrayofarrays;
    _arrayofarrays = new double[x * y];
    if (_arrayofarrays)
        std::copy(other._arrayofarrays, other._arrayofarrays + (x * y), _arrayofarrays);
}

CMatrix& operator =(const CMatrix& rval) {
    delete [] _arrayofarrays;
    _arrayofarrays = new double[ rval.x * rval.y];
    std::copy(rval._arrayofarrays, rval._arrayofarrays + (rval.x * rval.y), _arrayofarrays);
    x = rval.x;
    y = rval.y;
    return *this;
}



double *_arrayofarrays;
size_t x;
size_t y;
};

编辑:

实际上现在我已经意识到,在运行这部分代码之后崩溃了。在调用此代码之前,我有一个类的实例,我们称之为a,b,c,然后我想设置a = b-c; 这是第一次运作良好...但是当我想重复它,然后崩溃

CMatrix CMatrix::operator-(const CMatrix &matrix) const {
if (this->x != matrix.x || this->y != matrix.y) {
    throw CSizeException(matrix.y, matrix.x, this->y, this->x, '+');
};
CMatrix m(this->x, this->y);
CMatrix l(matrix.x, matrix.y);
l._arrayofarrays = this->_arrayofarrays;
CMatrix o(matrix.y, matrix.y);
o = matrix;


CMatrix result(this->x, this->y);
for (unsigned int i = 0; i < this->x; i++)
    for (unsigned int j = 0; j < this->y; j++)
        m[i][j] = l[i][j] - o[i][j];           
return m;
}

2 个答案:

答案 0 :(得分:4)

这样的东西?

~CMatrix() {
    delete[] _arrayofarrays;
}

所有分配有new[]的数组必须通过匹配调用delete[]来销毁。此外,您可以从复制构造函数中删除delete[]语句:

CMatrix(const CMatrix& other) : _arrayofarrays(), x(other.x), y(other.y) {
    // delete [] _arrayofarrays;
    // ^^^^^^^^^^^^^^^^^^^^^^^^^
    // This is unnecessary: you are creating a new object, so this data member
    // is not pointing to any previously allocated array

    // ...
}

<强>更新

  

来自评论)我无法使用此代码... _arrayofarrays实际上是2D数组,因此导致程序运行失败..可能是段错误

这是不正确的:_arrayofarrays是一维数组,调用delete[]是正确的方法来销毁它。如果这样做会导致段错误,可能你在其余的代码中做错了。

作为建议,通过原始指针newdelete(或其数组对应物)避免手动内存管理通常是一个好主意,因为它容易出错,很容易导致内存泄漏或无效指针/引用的解除引用。

请考虑使用标准容器,例如std::vector<>std::deque<>


修改

在你正在做的operator -代码中:

l._arrayofarrays = this->_arrayofarrays;

这样你就有了两个矩阵对象,它们封装了同一个数组:因此,删除其中一个也会使另一个无效。这可能是您问题的根本原因。

另外,你在那里创造了太多的临时工。除非我遗漏了什么,否则这应该足够了:

CMatrix CMatrix::operator-(const CMatrix &matrix) const {
    if (this->x != matrix.x || this->y != matrix.y) {
        throw CSizeException(matrix.y, matrix.x, this->y, this->x, '+');
    };

    CMatrix result(this->x, this->y);
    for (unsigned int i = 0; i < this->x; i++)
        for (unsigned int j = 0; j < this->y; j++)
            result[i][j] = (*this)[i][j] - matrix[i][j];

    return result;
}

答案 1 :(得分:2)

在析构函数中删除内存的方法与在其他成员函数中编写内存的方法相同:

    delete [] _arrayofarrays;

如果崩溃那么你必须在其他地方破坏内存。尝试使用valgrind检查内存错误。

您的复制构造函数存在一些问题:

CMatrix(const CMatrix& other) : _arrayofarrays(), x(other.x), y(other.y) {
    delete [] _arrayofarrays;

这没用,没有什么可以删除的。

    _arrayofarrays = new double[x * y];
    if (_arrayofarrays)

这个if没用,new将抛出异常或返回非空指针,因此检查它是否为空是没用的。

        std::copy(other._arrayofarrays, other._arrayofarrays + (x * y), _arrayofarrays);
}

您的赋值运算符也存在一些问题:

CMatrix& operator =(const CMatrix& rval) {
    delete [] _arrayofarrays;
    _arrayofarrays = new double[ rval.x * rval.y];

如果new分配抛出异常然后让_arrayofarrays持有悬空指针,则应该在删除旧内存之前分配新内存或者将旧内存归零使用delete[]

后的指针