前言
熟悉.Net开发的朋友会知道,项目在编译时,除了可执行的.exe文件外,在运行目录常常会生成.pdb的文件,好多人都不知道这个文件是干什么用的,今天就详细介绍一下。
介绍
本文帮助那些处于初级或中级水平的开发人员,但是他们对PDBs的重要性以及为什么需要它们没有太多的理解。
什么是PDB
PDB是程序数据库文件的首字母缩写。
PDB文件通常是在编译期间从源文件创建的。它存储模块所有符号的列表,其中包含它们的地址,可能还有文件的名称和声明符号的行。
为什么PDB是一个单独的文件?
这些符号本来可以很容易地嵌入到二进制文件中,但反过来又会使二进制文件的大小变得更大(有时是几兆字节)。为了减少文件的大小,现代编译器和早期的大型机调试系统将符号信息输出到一个单独的文件中,对于Microsoft编译器,这个文件称为.PDB文件。
PDB文件包含什么?
以下是PDB文件存储的一些重要信息:
局部变量名——为了证明pdb包含局部变量名,我们将利用反射器将其pdb在与程序集相同的文件夹中进行反编译。反射工具有一个选项,称为“显示PDB符号”,如屏幕截图中所示,当检查时也加载相应的PDB用于该程序集。当您检查选项,你可以看到反编译的代码相同的变量名,你的实际代码,但在缺乏PDB或未经检查该选项时,局部变量反编译的代码中,字符型变量名会被替换成“str”,数值型的则是“num”等等。
源文件名字
源的行号。
源索引(后面部分解释)
要显示PDB包含源文件名称和源文件的行号(第2和第3),首先在相同的文件夹中运行以下控制台应用程序,然后删除PDB文件。
namespace UnderstandingPDBs
{
class Program
{
static void Main(string[] args)
{
try
{
int sum = Add(5, 10);
decimal value = Divide(10, 0);
}
catch
{
}
}
private static int Add(int i, int j)
{
return i + j;
}
private static decimal Divide(int i, int j)
{
try
{
return i / j;
}
catch (Exception ex)
{
LogError(ex);
throw ex;
}
}
private static void LogError(Exception ex)
{
using (var txtWriter = new StreamWriter(@"dump.txt",true))
{
string error = "Exception:" + ex.Message +
Environment.NewLine + "StackTrace:" + ex.StackTrace
if(ex.InnerException!=null)
error=error+"INNER EXCEPTION:"+ex.InnerException;
txtWriter.WriteLine(error);
}
}
}
}
如果目录中存在PDB文件,下面是应用程序抛出的异常:
Exception:Attempted to divide by zero. StackTrace: at UnderstandingPDBs.Program.Divide(Int32 i, Int32 j) in C:UsersRishiDocumentsVisual Studio 2010ProjectsUnderstandingPDBsProgram.cs:line 33
如果没有PDB文件,异常信息如下:
Exception:Attempted to divide by zero. StackTrace: at UnderstandingPDBs.Program.Divide(Int32 i, Int32 j) ---------
显然,带有PDB的文件显示了异常被抛出的类的行号和文件名。
调试器如何加载PDB ?
Visual Studio调试器假设PDB文件位于与DLL或EXE相同的文件夹下。每一次程序集编译生成唯一的PDB文件,这意味着,即使没有代码改变,也不能使用在以前的编译中创建的PDB文件。调试器通过将PDB中的特定GUID与二进制的GUID进行比较,发现PDB是否跟二进制文件相匹配。这个Guid在编译过程中嵌入到二进制和PDB中,并将PDB与二进制文件紧密连接。
Visual Studio中不同的Build设置。
Visual Studio有3种不同的Build选项,可以控制调试符号的生成:
none:PDB文件将不会生成。
pdbonly:调试符号只能在PDB文件中,而不是二进制文件中。
Full:与PDB二进制中的符号一起也包含一些调试符号。
Full是Visual Studio中设置的默认选项。
参考MSDN文档:
如果您使用/调试:full,请注意,JIT优化代码的速度和大小会受到一定的影响,并且对代码质量的影响很小。我们建议/调试:pdbonly或没有PDB来生成发布代码。
我们应该和二进制文件一起部署PDBs吗?
如果交付件的大小不是问题,最好将PDB和其他二进制文件一起部署,因为它有助于提供更多关于异常的信息,就像我们在上面的例子中看到的那样。这些PDBs对于某些用户在某些情况下会非常有用,因为某些用户没有PDB会使生活变得困难。
这并不是说您必须拥有PDBs和二进制部署来获得关于异常的额外信息。同样可以使用符号服务器和源索引来实现,我将在下面的主题中讨论。
和PDB安全风险吗?
任何使用DLL/EXE的人都可以很容易地进行反向工程,使用诸如反射器之类的工具来生成带有或不带PDB的源代码。因此,在这种情况下,不提供PDB将不会有多大帮助。
如果PDB被部署,并且用户无法访问二进制文件,那么向他们显示堆栈跟踪信息并让他们知道应用程序的内部信息不是一个好主意。
Symbol Server
符号服务器
符号服务器用来存储被调试器所知道的pdb文件,可以用来查找更详细的调用堆栈信息。
我们可以使用symstore.exe设置自己的符号服务器,它允许调试器找到与二进制相关的实际PDB。symstore.exe包含在窗口包的调试工具中。
微软还保留了符号服务器,我们可以通过从微软的符号服务器加载PDBs来使用它。
如何以及为什么加载微软符号存储?
当您在debug点和open Modules窗口停止执行时(如下所示),您将会发现所有的dll(外部或内部)加载到该断点之前,但是默认情况下的符号状态将显示“无法找到或打开pdb文件”,除了您的pdb。这些是Microsoft BCL二进制文件,因为我们的调试器找不到相关的PDBs,所以没有加载。
要加载这些符号,可以去Debugging->Symbols ,并检查微软的符号服务器,并将缓存符号作为任何共享文件夹在这个目录中提供,以便所有的开发人员都可以使用它。
由于这些二进制文件是您的应用程序之外的,所以您还需要在Debugging->General 菜单中取消 “Enable just my Code” 。
在下面的截图中,您可以看到我已经加载了符号,现在符号的状态显示了“Symbols loaded”。
这怎么有用呢?
您可以在代码中放置断点,并在没有加载符号的情况下查看调用堆栈。
下图显示了没有加载符号的调用堆栈,它只是将我的方法和BCL的方法显示为[外部代码]。
在加载了这些符号之后,调用栈将在断点之前显示所有的方法调用(参见下图)。当我们想知道调用的外部方法是什么时,它当然是有帮助的,这样它就可以使用反射器或调试拆解来分析,而在我们的应用程序中,由于外部代码而导致某些行为更改的任何特定问题。
与符号服务器一样,也有称为源服务器的东西,用于检索用于构建任何特定应用程序的源文件的确切版本。在构建时,二进制文件可以被索引,并且该信息存储在PDB文件中,它帮助源服务器找到确切的源文件。
您可以检查MSDN了解更多关于符号和源存储。
PDB文件是微软专有格式的文件,也没有提供什么文档详细介绍。因为微软并未公布PDB内部细节,所以对于这个文件一直是一个迷。
1 宝宝出生后,对周围的环境一无所知。他还不能自主地控制身体和思想,他不会说话表达。所以对他来说,只有玩耍才能使他完全忘我地把注意力集中到环境中的他感兴趣的事物上,并运用自己的肢体、感官、大脑去探索它,不断重复,直到他完全掌握才转向下一个目标。渐渐地,他的运动能力增强了,感受能力丰富敏锐了,思维能...
前言: 各位小伙伴大家好,这里是不仅游戏打得6,长得还很帅的熊猫哥哥~ 王者荣耀中最有价值的游戏道具之一就是荣耀水晶,玩家们可以使用荣耀水晶来兑换特定商店中的皮肤和英雄,例如武则天、小乔-天鹅之梦等等,有很多高人气皮肤。 荣耀水晶需要通过积分夺宝来获取,非常看运气,很多手气...
微信在我们生活中扮演的社交角色是不可替代的,我们在上班、生活等各地都已经离不开他了,电脑端和手机端的互动已是广泛应用于生活中和工作学习生活中。比如互传文件、图片等等这样即方便又节省很多步骤,省时省力。 但是我们很多人在生活中都不止一个微信号,大家都知道电脑端微信只能登录一个微信...
随着互联网的高速发展,企业网站建设早已成为企业运营的一种必要手段。企业拥有一个自己的官网是非常重要的,也是一个企业互联网战略的基础。任何一个企业都应该有一个属于自己的网站,这个网站 随着互联网的高速发展,企业网站建设早已成为企业运营的一种必要手段。企业拥有一个自己的官网是非常重要的,也是一个企业互...
至于为什么加盟骗局加盟骗局这么说?这和加盟的大环境有很大的关系。加盟是由公司先创立实体店,进行营销,达到招商时限后。由公司发布广告吸引加盟商,加盟商考察招商公司后,和招商公司签署加 至于为什么加盟骗局加盟骗局这么说?这和加盟的大环境有很大的关系。加盟是由公司先创立实体店,进行营销,达到招商时限后。...
最近美国大选轰轰烈烈,新闻里总有白宫幕僚这个词出现,随着所追捧的候选人胜者为王,大选期间作为幕僚参与竞选团队的人也能够得到白宫里的一席之地,那么幕僚这个职位在中国古代又是什么发展路 最近美国大选轰轰烈烈,新闻里总有白宫幕僚这个词出现,随着所追捧的候选人胜者为王,大选期间作为幕僚参与竞选团队的人也能...
哈喽,各位小伙伴,今天给大家分享下手机的字体在哪里设置,这里边的字体一般包括苹果系统字体和安卓系统的字体设置。苹果系统字体大小设置一般来说系统字体大小设置的步骤都差不多,在这里我带 哈喽,各位小伙伴,今天给大家分享下手机的字体在哪里设置,这里边的字体一般包括苹果系统字体和安卓系统的字体设置。 苹...
“您好,这里是白银市公安局反诈骗中心,请问您最近有没有接到过诈骗电话或信息,让您告知手机验证码、点击陌生链接、扫陌生二维码,或者以各种理由要求您转账汇款的……”近日,记者来到白银市打击治理电信网络新型犯罪违法犯罪中心,看 “您好,这里是白银市公安局反诈骗中心,请问您最近有没有接到过诈骗电话或信息,...
iPhone14promax在官方直营店依旧没现货,包括京东或者天猫旗舰店,想买现货的朋友们在官网蹲到了吗?目前,iPhone14promax黄牛加价1200,可以自己卖赚点钱,如果自己加钱买的话就没必要了。iPhone iPhone14promax在官方直营店依旧没现货,包括京东或者天猫旗舰店,...
红薯含有丰富的维生素、矿物质以及膳食纤维等,适量食用对身体有一定的好处,很多人都会在家蒸红薯吃,蒸红薯一般都是要清洗干净的,洗干净的红薯最好尽快蒸熟吃,这样味道最美味,也可以放在第二天蒸,一般只要不去皮,都是可以隔天蒸煮 红薯含有丰富的维生素、矿物质以及膳食纤维等,适量食用对身体有一定的好处,很多...
客厅里的红木家具能不能随便摆是很多人好奇的,毕竟客厅作为家中比较重要的场景之一试,大家对该事的每个节点都尤为的关注,下面小编就和大家一起看看红木家具怎么摆放在客厅最好。客厅里的红木家具能不能随便摆不能。现在的社会,大家的 客厅里的红木家具能不能随便摆是很多人好奇的,毕竟客厅作为家中比较重要的场景之...
燕麦保存方法和注意事项,怎么储存麦片。小编带你了解更多相关信息。 1、燕麦的储存方法,选购杂粮燕麦片时,尽量选择单独包装的小包,不要买大包装的。如果买的是大袋,每次吃完记得要用夹子夹紧袋口;或者买有真空密封锁扣的袋装。 2、麦片不要放在潮湿的地方,最好放在可以密封的袋子或者罐子里;更不要放置...
参数上的不同。 操作方法 01红米Note拥有5.5英寸大屏,相比4.7英寸的红米手机和红米1S电信版屏幕更大 02小米红米 1S电信版手机性能上还是可以和红米Note媲美的。 03像素更为出色 04小米红米 池,相比红米手机和红米1S内置的2000mAh容量电池更为出众...
雪莲果是比较特别的一种水果,它含有多种营养成分,适量食用对身体有一定好处,很多人都喜欢吃雪莲果。雪莲果吃了可以润肠通便、补充营养、生津止渴、清热降火等,还可以促进消化吸收等,但是一次不能吃太多,以免引起肠胃不适。雪莲果的 雪莲果是比较特别的一种水果,它含有多种营养成分,适量食用对身体有一定好处,很...
柚子是秋季比较常见的一种水果,它的果肉比较饱满,吃起来鲜嫩多汁,味道很不错,营养价值高,深受人们喜欢。虽然柚子好吃,但是一次也不能吃太多,一个柚子的热量相当于1.3碗米饭热量,吃多了也会容易长胖的。一个柚子的热量相当于多 柚子是秋季比较常见的一种水果,它的果肉比较饱满,吃起来鲜嫩多汁,味道很不错,...