stack overflow异常分析及解决办法

  stack overflow异常是程序中常常会碰到的,究其原因,是进程空间中的栈不够用。windows默认栈大小是1M,使用的栈超过了这个范围就会包stack overflow异常。

  产生原因可以归结为两类:

  1. 错误。程序中出现了不该出现的死循环,例如递归函数没有出口,这种情况,不管栈空间有多大,早晚要溢出的。这类错误,通过debug,比较容易查找。举个例子:
long func(int n)
{
  return n*func(n-1);
}


这就是个没有出口的递归函数,只要该函数被调用,必然在某一次执行该函数时会引起栈溢出,具体是第多少次调用不好说。但是这个错误只要修改了,就可以纠正:

long func(int n)
{
  if(n==1) return 1;
  return n*func(n-1);
}

      2. 确实栈不够使用。保存到栈上的主要内容是:局部变量和函数调用的函数环境(包括函数参数等)。首先,对于局部变量,程序员应该比较清楚,例如:

char buff[1024*1024]

            buff大小为1M,如果你的栈默认大小也是1M,这时就会发生栈溢出(因为其他东西还要占掉少量栈空间,局部变量能用的空间肯定小于1M),在这种情况下,程序执行到main函数之前,就会跳出stack overflow异常。

另一种情况函数调用深度太深,这种情况一般发生在递归调用中,比如说一个求斐波那契数列的递归函数:

long func(long n)
{
  if(n<=0) return -1;
  if(n==1) return 1;
  if(n==2) return 1;
  return func(n-1) + func(n-2);
}


如果你调用func(1024*1024); 肯定就要栈溢出。

当栈不够使用时,一种办法是修改程序,看程序有没有可以优化的地方,例如,使用动态规划来代替递归算法等。另一种方法是增大栈空间,在vs2010中设置,项目->属性->连接器->系统->栈保留空间,如图。

 

总结,栈一不会使用得很大,主要还是要注意递归调用时,引起的栈溢出,多数情况可以通过算法优化来解决。递归深度不建议太深。

### RT-Thread 主栈溢出问题分析 在嵌入式操作系统中,栈空间通常有限,容易发生栈溢出错误。对于 RT-Thread 操作系统而言,主栈(main stack)主要用于启动内核和其他初始化操作。 当遇到主栈溢出时,常见的表现形式包括但不限于: - 系统崩溃或重启 - 不明原因的异常中断 - 数据损坏或不一致行为 #### 栈溢出的原因 1. **局部变量过大** 如果函数内部声明了过大的数组或其他数据结构,则可能导致当前调用帧超出分配给该线程堆栈的空间限制[^1]。 2. **递归深度过高** 过深的递归调用会持续消耗栈资源直至耗尽可用内存区域。 3. **动态分配失败** 使用 `alloca()` 或者其他方式尝试临时增加栈大小也可能引发此类问题。 4. **配置不当** 初始化阶段未合理设置初始栈尺寸参数也会间接造成此现象。 #### 解决方案 针对上述情况,可以采取如下措施预防和修复主栈溢出: ##### 调整栈大小 通过修改编译选项或链接脚本来增大默认栈容量是一个简单有效的方法。例如,在 GCC 编译器环境下可以通过 `-Wl,--stack,<size>` 参数指定新的栈长度;而在 Keil MDK 中则可以在项目属性里调整 Stack Size 字段值。 ##### 减少大对象占用 尽可能避免在函数体内创建大型静态/自动存储期的对象实例,转而考虑采用全局变量、静态成员或是借助 heap 动态申请所需缓冲区。 ##### 防止过度递归 仔细审查代码逻辑,确保不会因为意外条件触发无限次自我调用循环。必要时引入计数器机制加以控制最大迭代次数。 ##### 启用检测功能 部分平台支持硬件辅助监控手段如 ARM Cortex-M 的 MPU (Memory Protection Unit),能够实时捕捉非法访问事件并及时终止潜在风险进程。 ```c // 设置MPU保护范围以防止越界写入 void setup_mpu(void){ // ...具体实现... } ``` 另外还可以启用软件层面的安全防护策略比如 ASAN(AddressSanitizer)[^3], 它能在运行期间标记那些被怀疑有问题的地方从而帮助定位根本原因所在。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值