__purecall 链接错误

原文链接:http://www.titilima.cn/show-548-1.html

不知道诸位有没有过这样的经历:本是简单合法的 C++ 代码,但编译链接的时候却出现了如下的链接错误:

> error LNK2001: 无法解析的外部符号 __purecall

在解决这个问题之前,我们可以一起重现这个错误,先。新建一个 Win32 工程,打开 VS 的工程设置,修改如下项目:

  1. 将“启用 C++ 异常”设为“否”;
  2. 将“基本运行时检查”设为“默认值”;
  3. 将“缓冲区安全检查”设为“否(/GS-)”;
  4. 将“启用运行时类型信息”设为“否(/GR-)”;
  5. 将“忽略所有默认库”设为“是(/NODEFAULTLIB)”。

然后,在源文件中输入如下的代码:

我来解释一下:修改工程设置实质上是剥离了工程对 VS 默认 CRT 的依赖,因此这里再使用的 C++ 内容就必须都要自己实现,包括 new、delete 以及程序的入口——也就是 WinMainCRTStartup。在代码输入完成后,构建这个工程,就可以如愿得到本文开头的那个链接错误了。

虽然工程并未构建成功,不过你仍然可以用反汇编工具打开编译器生成的 obj 文件,秘密就藏在其中。下面我只是简单说明一下 B 对象的构造过程,不列汇编代码了就。

  1. 调用 new 申请内存空间。
  2. 调用 A::A,使用 A 类的虚表来初始化 A 类的子对象。
  3. 调用 B::B,使用 B 类的虚表来初始化对象。

问题就出现在第 2 步:A 类是个纯虚类,那么它的虚表是什么样子的?我在 obj 文件中找到了这个表,如下。

public ??_7A@@6B@
; const A::`vftable'
??_7A@@6B@      dd offset __purecall

如你所见,虽然 A::foo 是个纯虚函数,但编译器仍然为之准备了虚表中的一个栏位,只不过这个栏位的内容被一个名为 __purecall 的符号代替了。由于 __purecall 这个符号存在于默认的 CRT 之中,但我们先前又剥离了对 CRT 的依赖,因此在链接的时候就会出现 LNK2001 的链接错误。

解决方法很简单,再准备一个简单的 _purecall 就可以了:

当然,其中最好加个断言什么的,以示这是一个不应到达的区域。

此外,VS 的 C++ 编译器对此种情况提供了特殊的支持,也就是使用 __declspec(novtable) 来定义无虚表的纯虚类。考虑如下代码:

这样亦能解决这个问题。

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值