__purecall 链接错误

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

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

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

将“启用 C++ 异常”设为“否”; 
将“基本运行时检查”设为“默认值”; 
将“缓冲区安全检查”设为“否(/GS-)”; 
将“启用运行时类型信息”设为“否(/GR-)”; 
将“忽略所有默认库”设为“是(/NODEFAULTLIB)”。 
然后,在源文件中输入如下的代码:

view plaincopy to clipboardprint?
#include <Windows.h>   
#include <tchar.h>   
  
void* operator new(size_t _Size)   
{   
    return HeapAlloc(GetProcessHeap(), 0, _Size);   
}   
  
void operator delete(void* ptr)   
{   
    HeapFree(GetProcessHeap(), 0, ptr);   
}   
  
class A   
{   
public:   
    virtual void foo(void) = 0;   
};   
  
class B : public A   
{   
public:   
    void foo(void)   
    {   
        OutputDebugString(_T("Hello, World!\r\n"));   
    }   
};   
  
extern "C" void WinMainCRTStartup(void)   
{   
    A* p = new B;   
    p->foo();   
    delete p;   
}  
#include <Windows.h>
#include <tchar.h>

void* operator new(size_t _Size)
{
    return HeapAlloc(GetProcessHeap(), 0, _Size);
}

void operator delete(void* ptr)
{
    HeapFree(GetProcessHeap(), 0, ptr);
}

class A
{
public:
    virtual void foo(void) = 0;
};

class B : public A
{
public:
    void foo(void)
    {
        OutputDebugString(_T("Hello, World!\r\n"));
    }
};

extern "C" void WinMainCRTStartup(void)
{
    A* p = new B;
    p->foo();
    delete p;
}

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

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

调用 new 申请内存空间。 
调用 A::A,使用 A 类的虚表来初始化 A 类的子对象。 
调用 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 就可以了:

view plaincopy to clipboardprint?
extern "C" int __cdecl _purecall(void)   
{   
    return 0;   
}  
extern "C" int __cdecl _purecall(void)
{
    return 0;
}

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

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

view plaincopy to clipboardprint?
class __declspec(novtable) A   
{   
public:   
    virtual void foo(void) = 0;   
};  
class __declspec(novtable) A
{
public:
    virtual void foo(void) = 0;
};

这样亦能解决这个问题。
posted on 2010-06-01 23:59  carekee  阅读(1354)  评论(0编辑  收藏  举报