当数组大小太大时,C ++程序停止工作

时间:2013-04-20 18:39:33

标签: c++ visual-c++ visual-c++-2010

我在MVC ++ 2010中用c ++编写了一个代码。在其中,程序迭代一维指针数组(double *)的元素。但是,当我使输入(指针数组的大小)非常大(例如15000)并运行程序时,它会停止工作并显示一个关闭程序的窗口,因为它没有响应!有什么问题?

以下是构建我正在讨论的数组的代码的一部分:

map<int, double *> CF;
CoefficientMap(CF);

double *T = new double[I * J];
for (int r = 1; r <= I * J; ++r)
    T[r] = 100;
SOR(T, CF, 1.8);

这里是迭代器函数:

void SOR(double *T, map<int, double *> &CF, double w)
{
int iter = 0;
cout << "Stage 2: Solving the linear system of equations using SOR method... ";

const double tol = 0.00001;
double error = tol + 1;

double *TOld = new double[I * J];
for (int i = 1; i <= I * J; ++i)
    TOld[i] = 100;

while (abs(error) > tol)
{
    ++iter;
    for (int i = 1; i <= I * J; ++i)
        T[i] = (CF[i][0] + CF[i][1] * T[i + 1] + CF[i][2] * T[i + J] + CF[i][3] * T[i - J] + CF[i][4] * T[i - 1]) * w + (1 - w) * T[i];

    error = errorCalc(TOld, T, I * J);

    for (int i = 1; i <= I * J; ++i)
        TOld[i] = T[i];

    if (iter % 100 == 0)
    {
        cout << endl << endl;
        cout << "100 iterations done, please wait..." << endl << "Total accumulative error till this point: " << error << endl;
    }

    if (iter > 10000)
        return;
}

cout << "Done!" << endl << endl;
cout << "Converged after " << iter << " iterations!" << endl;
cout << "Final accumulative error: " << error << endl << endl;

}

现在,当(I * J)变得足够大(例如15000)时,程序停止工作!

3 个答案:

答案 0 :(得分:3)

最有可能的解释是,你的堆栈空间不足。简单的解决方法是使数组静态或全局。您也可以从堆中使用new进行分配。两者都将数组移出堆栈。

最好的可能是使用智能指针并将其放入堆:

std::unique_ptr<double[]> arrayOfDoubles(new double[size]);

当智能指针变量超出范围,无需手动删除时,将负责释放内存。

要获得更好的答案,请编辑问题以包含代码...


您添加的代码至少存在数组索引问题。索引从0开始,然后转到数组大小减1。正确的循环:

double *T = new double[I * J];
for (int r = 0; r < I * J; ++r)
    T[r] = 100;

你在其他循环中也有同样的错误,同样的修复。

替代修复:如果你想从1开始索引(比如,因为你有这样编写的伪代码算法并且不想改变索引),最简单的是分配一个更大的数组而不使用索引0:

double *T = new double[I * J + 1];

这样你就可以使用当前的循环。


一个数组元素的缓冲区溢出是令人讨厌的,因为在分配的内存块的末尾通常可能有未使用的空间,因此在更改数组大小并且未使用的空间消失之前,bug可能会完全被忽视。即使溢出导致堆损坏,也可能会被忽视,直到您更改代码和损坏更改的效果。因此,例如,如果您不幸,添加调试代码可能会隐藏问题。

答案 1 :(得分:2)

听起来你正在堆栈上分配一个普通数组,如下所示:

void f()
{
    double a[123456];
    ...
}

堆栈的大小有限 - 您应该使用new分配,或者(更好)使用std::vector

答案 2 :(得分:2)

您在堆栈上分配了太多空间,因此没有足够的内存来满足您的请求。作为替代方案,您可以为对象提供静态存储持续时间,或者您可以使用new将其放在免费商店中:

std::unique_ptr<int[]> ptr(new int[size]);