如何在矩阵类中释放内存

How can I free memory in Matrix Class?

本文关键字:释放 内存      更新时间:2023-10-16

我有一个矩阵类,具有以下构造函数:

template<class T>
Matrix<T>::Matrix(unsigned rows, unsigned cols) :
        rows(rows), cols(cols) {
    index = 0;
    data_ = new T[rows * cols];
}
template<class T>
Matrix<T>::~Matrix() {
    delete[] data_;
}

当我计算矩阵的逆运算时,我想释放名为 a 的临时变量:

template<class T>
Matrix<T> Matrix<T>::inverse() {
    unsigned i, j, k;
    Matrix<T> a(2 * rows, 2 * rows);        
    ....
    return tmp;
}

我以为这个变量会在函数结束时被破坏,但是当我测试时:

for (int i = 0; i < 3; i++) {   
        Matrix<double> m(5, 5);
        m << 5, 2, 4, 5, 6, 1, 3, 1, 2, 5, 2, 5, 2, 7, 2, 9, 2, 1, 0.1, 0.43, 1, 0, 0, 0, 1;
        m.inverse();
        std::cout << m << std::endl;
    }

在第一个循环中,a用零初始化,但下一步a的初始值是以前的值,所以a(k+1)=a_endvalues(k) .为什么会这样?

内存确实被释放了。然后在循环的下一次迭代中再次分配。并不是说您的Matrix构造函数不会将数组初始化为任何内容,因此它只包含恰好存储在现在属于它的内存中的内容。

在您的特定情况下,分配器可能会将第二次迭代a.data_放置在第一次迭代a.data_所在的同一位置,因此您仍然会看到旧值。

问题是您没有在构造函数中初始化动态分配数组的元素。要确保数组具有默认构造的元素或零初始化的元素,您需要在构造函数中执行此操作:

template<class T>
Matrix<T>::Matrix(unsigned rows, unsigned cols) :
        rows(rows), cols(cols) {
    index = 0;
    data_ = new T[rows * cols]();
                //             ^ HERE!
}

但正如评论中指出的那样,您可以使用std::vector<T>让您的生活更轻松:

template<class T>
Matrix<T>::Matrix(unsigned rows, unsigned cols) :
        rows(rows), cols(cols), data_(rows*cols) 
{ }

每次调用函数时,都会释放并重新分配内存。价值观的重叠只是偶然。新分配可能与旧分配位于同一位置。可以通过创建另一个参数化构造函数并传递单独的初始值设定项来测试这一点。

还可以通过编写输出语句来测试是否调用析构函数。

初始化数组与分配数组是分开的。

在析构函数中添加调试输出以验证是否已调用它。

同样要将数组初始化为零,可以使用内存集:

memset(a, 0, rows * cols * sizeof(T));

局部变量 'a' 被销毁,析构函数释放内存。但是,当再次分配内存时,内存位置具有先前的内容只是巧合。