最近写 CPP 项目遇到了一个问题,用了几个工具来解决,这里记录一下,和大家一起讨论。
1. 起因
我的一个 CPP 项目的 UI 框架使用的是 CefSharp,UI 层是 C#,而一些模块代码使用的是 CPP,运行报错如下
报错信息是
System.IO.FileLoadException:"未能加由"CefSharp.Core.Runtime.dl"导入的过程
第一感觉是进程加载某个配置文件或 dll 未成功,或者动态库版本问题,但错报不出来,而且 VS 的 Debugger 无法显示更多内容,用 Dependencies 看进程,dll 都是找到了的,那会是什么原因呢?
2. 分析
经过同事的提示,使用微软提供的 Windows Debugging Tools 调试套件 gflags.exe
工具,让进程加载过程中的日志能在 VS 的<输出>报出来,配置如下。在 Image File
中输入完整进程名,然后选中 Show loader snaps 显示加载器快照,然后重启进程。
然后重新运行,在 VS 的输出中就可以看到进程加载过程中的日志,如下。
在输出中搜 ERROR,可以看到加载哪些 dll 失败、警告等信息,有一些不重要,继续找发现加载的一个 dbghelp.dll 文件后,使用它的一个函数 SymGetSearchPathW
未找到失败,到库目录发现这个文件的时间比较老了,这个方法需要在 6.3 以上版本的 dbghelp.dll 文件才提供。
可以使用 dumpbin 命令的 dumpbin dbghelp.dll /EXPORTS
命令看到这个 dll 里并没有这个SymGetSearchPathW
方法,如下
也可以使用 Dependencies 这个工具来看 dll 中的入口函数,如下
3. 解决
将项目的库目录里老版本 dbghelp.dll 删掉,让进程去系统的库目录里找较新的 dll,然后运行成功。
4. 复盘
对于进程(也就是 Image File),gflags.exe
的 Show loader snaps 选项开启之后会允许捕获加载和卸载 dll 的信息,并在调试器控制台中显示这些信息,如果过程中有报错信息也可以一并展示,所以可以用来查找类似于加载卸载动态库报错的问题。这个工具还可以做很多事情,监控内存分配、检查内存泄露等等,非常实用,后面再找个时间学习一下。
Windows 上的 dumpbin
命令和 linux 下的 ldd
命令,是用来查看动态库信息的,包括可执行文件和动态链接库文件的依赖项。dumpbin
命令可以用来看动态库导出了哪些函数 dumpbin /EXPORTS a.dll
,exe 加载了哪些动态库 dumpbin /IMPORTS a.exe
,查看静态链接库中包含哪些函数 dumpbin /ALL /RAWDATA:none a.lib
。另外,这个命令默认是不在 cmd 里,安装了 VS 的话需要在 VS 的开发者命令提示 Developer Command Prompt 里运行才能找到这个命令。
网上的帖子大多深浅不一,甚至有些前后矛盾,在下的文章都是学习过程中的总结,如果发现错误,欢迎留言指出,如果本文帮助到了你,别忘了点赞支持一下哦,你的点赞是我更新的最大动力!~
PS:本文同步更新于在下博客 Github - SHERlocked93/blog 系列文章中,欢迎大家关注我的公众号 CPP下午茶
,直接搜索即可添加,持续为大家推送 CPP 以及 CPP 周边相关优质技术文,共同进步,一起加油~
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。