StackFrame和StackTrace在Unity和C#中的区别

本文为作者原创,转载请注明出处:https://www.cnblogs.com/zhaoqingqing/p/14050163.html



本文通过实际例子来看看StackFrame和StackTrace有什么区别,分别在.NET和Unity中测试。

.NET环境#

测试代码

Copy
using System; using System.Diagnostics; namespace ConsoleApplication1 { internal class Program { public static void Main(string[] args) { MyFunc1(); MyFunc2(); Console.ReadKey(); } static void MyFunc1() { Console.WriteLine(new StackFrame(true)); } static void MyFunc2() { Console.WriteLine(new StackTrace(true)); } } }

通过Debug或Release模式生成exe运行而非在IDE下运行,在这两种模式下有、无pbd文件,输出结果是一致的,如下所示

有pdb文件#

MyFunc1 at offset 59 in file:line:column E:\Code\csharp_study\ConsoleApplication1\ConsoleApplication1\Program.cs:15:34

在 ConsoleApplication1.Program.MyFunc2() 位置 E:\Code\csharp_study\ConsoleApplication1\ConsoleApplication1\Program.cs:行号 16
在 ConsoleApplication1.Program.Main(String[] args) 位置 E:\Code\csharp_study\ConsoleApplication1\ConsoleApplication1\Program.cs:行号 11

删除pdb文件#

MyFunc1 at offset 59 in file:line:column :0:0

在 ConsoleApplication1.Program.MyFunc2()
在 ConsoleApplication1.Program.Main(String[] args)

Unity环境#

我的unity版本 unity 2019.3.7 个人版

在Unity编辑器下#

测试代码如下:

Copy
void OnClickBtn1() { MyFunc1(); MyFunc2(); } void MyFunc1() { Debug.Log(this.name + "click stacktrace:\n" + new StackTrace(true) + "\n"); } void MyFunc2() { Debug.Log(this.name + "click stackframe:\n" + new StackFrame(true) + "\n"); }

new StackFrame(true)打印的堆栈如下:
Canvasclick stackframe:
MyFunc2 at offset 1 in file:line:column E:\Code\UGUIDemo\Assets\Function\TestLogStackTrace.cs:58:9

new StackTrace(true) 打印的堆栈(在上面的基础上会有更多的Unity底层堆栈)

Canvasclick stacktrace:
at TestLogStackTrace.MyFunc1 () [0x00001] in E:\Code\UGUIDemo\Assets\Function\TestLogStackTrace.cs:53
at TestLogStackTrace.OnClickBtn1 () [0x00001] in E:\Code\UGUIDemo\Assets\Function\TestLogStackTrace.cs:35
.....中间省略一些Unity的调用栈
at UnityEngine.EventSystems.EventSystem.Update () [0x000f9] in D:\Program Files\Unity\Editor\Data\Resources\PackageManager\BuiltInPackages\com.unity.ugui\Runtime\EventSystem\EventSystem.cs:377
UnityEngine.Debug:Log(Object)
ButtonScene:OnClickBtn1() (at Assets/uGUI/Scripts/ButtonScene.cs:40)
UnityEngine.EventSystems.EventSystem:Update() (at D:/Program Files/Unity/Editor/Data/Resources/PackageManager/BuiltInPackages/com.unity.ugui/Runtime/EventSystem/EventSystem.cs:377)

结论#

结合实践和反编译mscorlib.dll的代码来看

StackTrace中有一个StaceFrame列表,包含完整的调用栈,对于第二个函数的完整调用栈如下

Copy
new StackTrace(true) = {StackTrace} " 在 ConsoleApplication1.Program.MyFunc2()\r\n 在 ConsoleApplication1.Program.Main(String[] args)\r\n" FrameCount = {int} 2 Static members = {} Non-public members = {} frames = {StackFrame[]} Count = 5 [0] = {StackFrame} "GetStackFramesInternal at offset 0 in file:line:column <filename unknown>:0:0\r\n" [1] = {StackFrame} "CaptureStackTrace at offset 188 in file:line:column <filename unknown>:0:0\r\n" [2] = {StackFrame} ".ctor at offset 100 in file:line:column <filename unknown>:0:0\r\n" [3] = {StackFrame} "MyFunc2 at offset 27 in file:line:column <filename unknown>:0:0\r\n" [4] = {StackFrame} "Main at offset 43 in file:line:column <filename unknown>:0:0\r\n" m_iMethodsToSkip = {int} 3 m_iNumOfFrames = {int} 2

而stackFrame中只有当前函数的这一帧调用者信息

这个结论同时适用于.NET环境和Unity环境中。

目前我们Unity线上项目设置的堆栈为Script only

作者:赵青青   一名在【网易游戏】做游戏开发的程序员,擅长Unity3D,游戏开发,.NET等领域。
本文版权归作者和博客园共有,欢迎转载,转载之后请务必在文章明显位置标出原文链接和作者,谢谢。
如果本文对您有帮助,请点击【推荐】您的赞赏将鼓励我继续创作!想跟我一起进步么?那就【关注】我吧。
posted @   赵青青  阅读(886)  评论(0编辑  收藏  举报
编辑推荐:
· .NET Core 托管堆内存泄露/CPU异常的常见思路
· PostgreSQL 和 SQL Server 在统计信息维护中的关键差异
· C++代码改造为UTF-8编码问题的总结
· DeepSeek 解答了困扰我五年的技术问题
· 为什么说在企业级应用开发中,后端往往是效率杀手?
阅读排行:
· 10亿数据,如何做迁移?
· 推荐几款开源且免费的 .NET MAUI 组件库
· 清华大学推出第四讲使用 DeepSeek + DeepResearch 让科研像聊天一样简单!
· 易语言 —— 开山篇
· Trae初体验
历史上的今天:
2018-11-27 lua_local变量在new时不会被清空
点击右上角即可分享
微信分享提示
CONTENTS