如何避免堆栈溢出?

时间:2009-09-21 10:23:00

标签: c# .net .net-2.0

我使用CSharpCodeProvider编译我的代码,并在结果程序集中动态创建某个类的实例。比我叫一些方法。如果方法有递归,我得到StackOverflowException,我的应用程序终止。

我该如何避免这种情况?

using System;
using System.Runtime.Remoting;
namespace TestStackOverflow
{
    class Program
    {
        class StackOver : MarshalByRefObject
        {
            public void Run()
            {
                Run();
            }
        }

        static void Main(string[] args)
        {
        AppDomain domain = AppDomain.CreateDomain("new");

        ObjectHandle handle = domain.CreateInstance(typeof (StackOver).Assembly.FullName, typeof (StackOver).FullName);
        if (handle != null)
        {
            StackOver stack = (StackOver) handle.Unwrap();
            stack.Run();
        }

    }
}
}

相关:

  

What is a stack overflow?

7 个答案:

答案 0 :(得分:10)

StackOverflow表示您的递归过深,堆栈内存不足。例如:

public class StackOver
{
    public void Run() 
    { 
        Run(); 
    }
}

这将导致堆栈溢出,因为StackOver :: Run()将被反复调用,直到没有内存为止。

我怀疑在您的情况下,您可能错过了终止条件,或者您正在运行太多的递归迭代。

如果您尝试保持应用程序正常运行,请尝试:

namespace TestStackOverflow
{
    class Program
    {
        class StackOver : MarshalByRefObject
        {
            public bool Run()
            {
                return true; // Keep the application running. (Return false to quit)
            }
        }

        static void Main(string[] args)
        {
            // Other code...

            while (stack.Run());
        }

    }
}

答案 1 :(得分:8)

运行正在调用Run。那是无限递归。

    class StackOver : MarshalByRefObject
    {
        public void Run()
        {
            Run(); // Recursive call with no termination
        }
    }

答案 2 :(得分:1)

如果递归导致堆栈溢出,那么问题与编译类无关 - 递归函数需要终止条件,因为C# doesn't (usually) optimize tail calls

答案 3 :(得分:1)

避免使用递归函数的堆栈溢出的唯一方法是有一个明确的退出条件,无论输入如何,最终都会满足。您可以定义最大深度并在到达时停止进行递归调用,或者确保您检查的数据是有限的(并且在合理的限制范围内),或两者的组合。

答案 4 :(得分:0)

我没有CSharpCodeProvider的好背景,但我知道递归实现的算法可以通过循环实现

答案 5 :(得分:0)

确定。使用CSharpCodeProvider无关紧要。我在另一个域中使用Reflection加载程序集。我认为域名是出于安全原因而创建的。如何保护应用程序免于终止???

using System;
using System.Runtime.Remoting;
namespace TestStackOverflow
{
    class Program
    {
        class StackOver : MarshalByRefObject
        {
            public void Run()
            {
                Run();
            }
        }

        static void Main(string[] args)
        {
        AppDomain domain = AppDomain.CreateDomain("new");

        ObjectHandle handle = domain.CreateInstance(typeof (StackOver).Assembly.FullName, typeof (StackOver).FullName);
        if (handle != null)
        {
            StackOver stack = (StackOver) handle.Unwrap();
            stack.Run();
        }

    }
}
}

答案 6 :(得分:0)

每次从方法栏调用方法foo时,bar都会添加到调用堆栈中。调用堆栈用于在调用方法之前跟踪代码的位置,以便在foo完成时返回到那里。

以下递归函数

int Factorial(int n) {
    if (n == 0) { return 1; }
    return n * Factorial(n - 1);
}

在几次递归调用Factorial(5)之后,调用栈看起来像这样:

Factorial(5) -> Factorial(4) -> Factorial(3) -> Factorial(2) -> Factorial(1)

此时n为1,因此函数停止调用递归的情况,而是返回1.程序然后开始回调调用堆栈,整个事件返回120.

如果没有调用堆栈,程序在完成执行方法时将不知道返回的位置。

现在假设基本情况不存在,它看起来像这样:

int Factorial(int n) {
    return n * Factorial(n - 1);
}

在几次递归调用Factorial(5)之后,调用栈看起来像这样:

Factorial(5) -> Factorial(4) -> Factorial(3) -> Factorial(2) -> Factorial(1) -> Factorial(0) -> Factorial(-1) -> Factorial(-2) -> Factorial(-3) -> Factorial(-4) -> Factorial(-5) -> Factorial(-6) -> Factorial(-7) -> Factorial(-8) -> Factorial(-9) -> Factorial(-10) -> Factorial(-11) -> Factorial(-12) -> Factorial(-13) -> Factorial(-14) -> Factorial(-15) etc…

因为没有代码停止调用自身的点,它将永远存在,并且调用堆栈将增长并增长并增长占用越来越多的内存,直到它超过已分配的内存和StackOverflow异常扔了。

有两种方法可以阻止这种情况发生,最好的方法取决于具体情况。

1提供基本案例。确保最终达到的某些条件会阻止函数调用自身。在Factorial的情况下,它是n == 1,但它可能已经过了一定的时间,它已经递归了一定次数,某些计算的某些结果在某些范围内,无论如何。只要它在堆栈太大之前停止重复使用。

2删除递归并重新写入。任何递归算法都可以重写为非递归算法。它可能不那么干净和优雅,但它可以做到。在阶乘论证中,它可能类似于:

int Factorial(int n) {
    int result = 1;

    for (int i = 0; i < n; i += 1) {
        result *= n;
    }

    return result;
}

如果目的是一次又一次地连续运行相同的函数,那么你可以重写递归

void Foo() {
    // Some code
    Foo();
}

作为

void Foo() {
    while (true) { // Some code }
}