写了多年代码,你会 StackOverflow 吗
写了多年代码,你会 StackOverflow 吗
Intro
准备写一个傻逼代码的系列文章,怎么写 StackOverflow
的代码,怎么写死锁代码,怎么写一个把 CPU 跑满,怎么写一个 OutOfMemory
的代码。
今天主要来看 StackOverflow
,StackOverlow
是一个著名的问答社区,做开发相关的应该都会知道这个网站,很多你遇到的问题都可以在这个网站上找到答案。
如何让你的代码发生 StackOverflow
在 C# 中有一个 StackOverflowException
的异常,发生 StackOverlow
也就是触发 StackOverlowException
,关于 StackOverflowException
的介绍可以参考微软文档
StackOverflowException
is thrown for execution stack overflow errors, typically in case of a very deep or unbounded recursion.
这里节选了一小段话, 在执行堆栈发生溢出错误的时候会抛出 StackOverflowException
,典型的案例就是一个特别深的或者没有边界的递归。
这里说明了一种典型的会发生 StackOverflowException
的示例就是递归,这种方式可能大家都知道了。
除此之外还有一种情况也会引发 StackOverflowException
,微软的文档上没有明确说明,我们知道,在 C# 中,值类型的内存分配一般是在栈上的,这个栈也就是线程栈,一个线程的栈是有限制的,这里可以出一个面试题,线程栈有多大
线程栈默认 1M,也是最大值,最大也是 1M(下次面试的时候可别不知道了哈)
详细介绍可以看文档 https://docs.microsoft.com/en-us/dotnet/api/system.threading.thread.-ctor?view=netcore-3.1
在线程栈上分配内存不够的时候同样也会引发 StackOverflowException
所以写出 StackOverflow
代码的两种方式就有了
- 死递归(没有边界,一直在循环调用自己)
- 线程栈内存分配
死递归
死递归这种方式最简单了,写一个方法调用自己即可,来看一个示例:
public static void Test1()
{
Test1();
}
通常这样的代码,大家一眼就能看出来,一般也不会写这样的代码,通常写出 StackOverflow
的代码都是有参数的,参数传递有问题所以导致的 StackOverflow
,比如下面这样的一个示例:
// StackOverflow
public static void Test2(int num)
{
if(num > 0)
{
Test2(num);
}
}
// work
public static void Test3(int num)
{
if(num > 0)
{
Test3(--num);
}
}
下面的代码是正常代码,上面的代码会 StackOverlow
,只是因为一行代码的失误导致的问题,这种情况更为常见一些
我之前也写过一个
StackOverflow
的代码,之前的代码也是带参数的递归,递归的参数是一个枚举,方法体里有多个枚举参数,结果在方法体递归的时候传错了参数,导致了一个死递归
线程栈内存分配
通常值类型的分配是在线程栈上的,但是我们也不会在一个线程上分配很多的值类型对象,所以通常也不会因为值类型的内存分配而导致 StackOverflowException
在 C# 7.2 之后引入了 Span
(.net core 2.1 之后默认包含了 Span
,.netstandard2.1
也包含了 Span
,如果是低版本或者 .net framework 需要额外引用一个包来支持 Span),我们可以在安全代码中使用 stackalloc
来实现在线程栈上分配内存,我们要使用这个来做测试,来看下面的测试代码:
public static void Test()
{
ReadOnlySpan<byte> bytes = stackalloc byte[1024 * 1024];
Console.WriteLine($"{bytes.Length} passed");
bytes = stackalloc byte[1024 * 1024 + 1];
Console.WriteLine($"{bytes.Length} passed");
}
可以看到在超过 1M 的时候就会发生 StackOverflow
Thread MaxStackSize
如果你比较细心,使用 Thread
比较多的话,你会发现 Thread
在实例化的时候,可以在构造方法里指定一个参数来指定线程的最大栈,参考文档:https://docs.microsoft.com/en-us/dotnet/api/system.threading.thread.-ctor?view=netcore-3.1
public Thread (System.Threading.ThreadStart start, int maxStackSize)
所以我们也可以指定一个比较小的 maxStackSize
来更容易测试出来 StackOverflow
来看一个示例:
public static void Test2()
{
var thread = new Thread(() =>
{
ReadOnlySpan<byte> bytes = stackalloc byte[16*1024+1];
Console.WriteLine($"{bytes.Length} passed");
bytes = stackalloc byte[32*1024+1];
Console.WriteLine($"{bytes.Length} passed");
bytes = stackalloc byte[256*1024+1];
Console.WriteLine($"{bytes.Length} passed");
}, 1);
thread.IsBackground = true;
thread.Start();
}
这里可以看到,我直接指定了 maxStackSize
为 1,但是实际测试下来并不是1,按微软的文档上所说,最小值是 256kb(如果我没理解错的话),但是实际测试下来并不是,没有到 256kb 就发生了 StackOverflow
(有大神看到还望指导一下),上面的测试输出结果
再来看一个递归的示例:
public static void Test3()
{
var thread = new Thread(() =>
{
TestMethod(1024);
}, 1);
thread.IsBackground = true;
thread.Start();
}
private static void TestMethod(int num)
{
if(num > 0)
{
num--;
TestMethod(num);
}
}
输出结果如下:
上面使用的数字也许不会在你的电脑上发生异常,你可以尝试自己调试一个临界值,或者直接设置一个比较大的值
More
StackOverflowException
不能通过 try...catch 来捕捉,当发生 StackOverflowException
时应用程序就会直接退出,通常线程栈上分配内存不够的情况基本上是不会发生的,所以对于怎么避免 StackOverflowException
我们只需要在写递归代码的时候小心一些,不要写成死循环递归就可以了。
Reference
- https://docs.microsoft.com/en-us/dotnet/api/system.stackoverflowexception?view=netcore-3.1
- https://docs.microsoft.com/en-us/dotnet/api/system.threading.thread.-ctor?view=netcore-3.1
- https://docs.microsoft.com/en-us/dotnet/csharp/language-reference/operators/stackalloc
- https://github.com/WeihanLi/SamplesInPractice/blob/master/StupidSamples/StackOverflowSample.cs
写了多年代码,你会 StackOverflow 吗的更多相关文章
- 不写1行代码,在Mac上体验ASP.NET 5的最简单方法
昨天微软发布了ASP.NET 5 beta2(详见ASP.NET 5 Beta2 发布),对ASP.NET 5的好奇心又被激发了. 今天下午在Mac OS X上体验了一下ASP.NET 5,而且借助Y ...
- 只写104行代码!在nopCommerce中如何实现自动生成网站地图
表告诉我说你不知道nopCommerce是什么.它是目前.NET中最流行的完全开源网上商城,由俄罗斯的团队在2008年开始立项一直开发到现在已经是3.3版本了.代码目前托管在codeplex上,有兴趣 ...
- 写出优美代码的两个方式:一步到位VS迭代优化
最近把手头这个安卓APP的所有事务性方法都写完了,有了以下体会,新手体会,老鸟轻拍 想写成优美代码的人一般都会有这样的想法: 一定要在写每一句代码,写每一个方法,构造每一个类的时候,都要记得优化: ...
- jQuery Validate 表单验证插件----通过name属性来关联字段来验证,改变默认的提示信息,将校验规则写到 js 代码中
一.下载依赖包 网盘下载:https://yunpan.cn/cryvgGGAQ3DSW 访问密码 f224 二. 添加一个另外一个插件jquery.validate.messages_cn.js. ...
- 程序员的恶性循环:加班->没空学习->老是写同等水平代码->无法提升代码质量->老是出BUG->老是需要修改->加班->...
程序员的恶性循环:加班->没空学习->老是写同等水平代码->无法提升代码质量->老是出BUG->老是需要修改->加班->...
- 将自己写的Python代码打包放到PyPI上
如果是开源的Python代码,为了能够让大家更方便的使用,放到PyPI上也许是个非常不错的主意(PyPI:Python Package Index).刚开始我以为要将代码打包放到PyPI上是一件非常复 ...
- 瞧一瞧,看一看呐,用MVC+EF快速弄出一个CRUD,一行代码都不用写,真的一行代码都不用写!!!!
瞧一瞧,看一看呐用MVC+EF快速弄出一个CRUD,一行代码都不用写,真的一行代码都不用写!!!! 现在要写的呢就是,用MVC和EF弄出一个CRUD四个页面和一个列表页面的一个快速DEMO,当然是在不 ...
- NASA关于如何写出安全代码的10条军规
博客搬到了fresky.github.io - Dawei XU,请各位看官挪步.最新的一篇是:NASA关于如何写出安全代码的10条军规.
- javascript 写一段代码,判断一个字符串中出现次数最多的字符串,并统计出现的次数
javascript 写一段代码,判断一个字符串中出现次数最多的字符串,并统计出现的次数 function test(){ var bt = document.getElementById(" ...
随机推荐
- 莫名其妙的Explain Plan
两张表的建表语句: CREATE TABLE hy_emp ( empno NUMBER(8,0) not null primary key, ename NVARCHAR2(60) not null ...
- [LeetCode]33. 搜索旋转排序数组(二分)
题目 假设按照升序排序的数组在预先未知的某个点上进行了旋转. ( 例如,数组 [0,1,2,4,5,6,7] 可能变为 [4,5,6,7,0,1,2] ). 搜索一个给定的目标值,如果数组中存在这个目 ...
- 福利来了~Linux一键部署包,环境安装不用愁!!!
前言 昨天一哥们的弟弟突然问我有没有部署过的Linux,公司连个运维都没有,服务器都要后端部署.... 你有没有相似的遭遇呢?公司规模小,后端即是运维,一份工资干两份活,哈哈~ 为了解决这老弟的困惑, ...
- hystrix(6) 命令执行
上一节中讲到了HystrixCommand有四种执行方法,这一节就来讲一下这四种方法直接的关系以及他们的实现. execute方法使用同步方式获取结果,本质是调用了queue方法获取了一个Future ...
- Spring Boot学习(二)搭建一个简易的Spring Boot工程
第一步:新建项目 新建一个SpringBoot工程 修改项目信息 勾选项目依赖和工具 选择好项目的位置,点击[Finish] 第二步:项目结构分析 新建好项目之后的结构如下图所示,少了很多配置文件: ...
- SSTI服务器模板注入(以及关于渲染,solt的学习)&&[BJDCTF2020]The mystery of ip 1
ssti服务器模板注入 ssti:利用公共 Web 框架的服务器端模板作为攻击媒介的攻击方式,该攻击利用了嵌入模板的用户输入方式的弱点.SSTI 攻击可以用来找出 Web 应用程序的内容结构. slo ...
- get请求传递json格式数据的两种方法
get请求参数为json格式数据,使用pyhton+request的两种实现方式如下: 方法一:使用requests.request() 示例代码如下: 1.导入requests和json impor ...
- 【PowerQuery】做了一万遍的工资条
前面已经了解了Excel.VBA.Python实现工资条,今天尝试用PQ做一遍 做之前迷惑了很久,如何能自定义长度 Table有Repeat函数,但是List没有.看来另外想办法 一步步接近目标 请 ...
- Lua设计与实现--读书笔记
目录 lua简介 一种通用的数据类型:lua_TValue 字符串 Table lua实现一个队列 lua简介 C++底层核心模块,暴露核心接口给lua脚本层,网络的收发都在c++层完成,本书简述lu ...
- 基础篇:java基本数据类型
1:java几种基本数据类型大小 关键字 类型 位数 (8位一字节) 取值范围(表示范围) byte 整型 8 -2^7 ~ 2^7-1 short 整型 16 -2^15 ~ 2^15-1 int ...