CLI / C ++链接错误2028和2019

时间:2011-04-13 11:10:16

标签: templates linker c++-cli

我在CLI项目中传递模板(我认为这是原因)时遇到了问题。

我有3层课程(出于封装原因和技术原因)。

我将展示一个有关错误部分的示例(没有关于封装原因的评论,这里没有展示它们)

A 位于C ++ DLL中:

class A {
public:
    template<class T>
    void foo(T blah) { //Do stuff }
}

Class B 包装A类(也是常规的非ref类):

class B {
public:
    template<class T>
    void foo(T blah) { a->foo(blah); }
private:
    A* a;
}

Class C 是一个ref类,它使用显式类型调用B类:

ref class C {
public:
    void foo(int blah) { b->foo(blah); }
private:
    B* b;
}

它们编译正常(.obj已创建),但链接器未正确链接对象。

我为该方法收到2个链接器错误:

  

错误LNK2028:未解析的令牌   (0A000645)“public:void __cdecl   B :: foo(类utils :: CustomString const   &amp;,int const&amp;)“   (?? $ foo的@ _N @乙@ @@命名空间$$ FQEAAXAEBVCustomString @ @@ utils的@ AEB_N Z)   在函数“private:void中引用   __clrcall C :: foo(int)“(?? $ foo @ _N @ Namespace @@ $$ FAE $ AAMXPE $ AAVString @ System @@ _ N @ Z)

     

错误LNK2019:未解析的外部   符号错误LNK2019:未解决   外部符号“public:void __cdecl   B :: foo(类int const&amp;)“   ?? $ foo的@ _N @乙@ @@命名空间$$ FQEAAXAEBVCustomString @ @@ utils的@ AEB_N Z)   在函数“private:void中引用   __clrcall C :: foo(int)“(?? $ foo @ _N @ Namespace @@ $$ FAE $ AAMXPE $ AAVString @ System @@ _ N @ Z)

修改

我现在没有和我在一起(不在同一台PC上),但它说它无法链接 C.foo <中引用的 B.foo / p>

我正在调试模式 / MDd 中使用 / clr 编译ref类(是的,它必须处于调试模式,因为其他依赖项都是编译的同样的方式)

有谁知道为什么会这样?更重要的是:如何解决这个问题?

修改

当设置要使用/ GL(整个程序优化)编译的B类(包装器)时,我得到一个不同的错误:
LNK2001

  

错误LNK2001:未解决   外部符号“public:bool __cdecl   Interface :: B :: foo(int&amp;)const“   (?? $ foo @ _J @ B @ Namespace @@ $$ FQEBA_NAEBVCustomString @ 123 @ AEA_J @ Z)

3 个答案:

答案 0 :(得分:1)

问题是visual studio的链接器的错误。

首先,您需要设置 / GL(整个程序优化)标志,以便在编译后链接模板。

然后你需要使用其中一种解决方法:

BUG: LNK2001 on Member Function When Use Nested Class Template

答案 1 :(得分:0)

这是C ++模板的标准损失,它们没有像.NET泛型那样的外部链接。您必须将模板函数定义放在头文件中,以便在包含ref类的源代码文件中#include它。就像你的代码片段一样。它不是特定于C ++ / CLI。

如果在没有/ clr选项的情况下编译其余的本机类(确实如此),请务必使用#pragma managed包装#include。像这样:

#pragma managed(push, off)
#include "FileWithTemplateDefinition.h"
#pragma managed(pop)

答案 2 :(得分:-1)

这些链接器错误指出您在某处缺少定义(而不是声明)。无法访问您的代码使我们难以确定,特别是没有任何冗长的链接器错误。

[编辑]

此代码在托管C ++项目中为我编译和链接:

#pragma managed

class A {
public:
    template<class T>
    void foo(T blah) { int i = 0; }
};

//Class B wraps class A (also regular non-ref class):
class B {
public:
    template<class T>
    void foo(T blah) { a.foo(blah); }
private:
    A a;
};

// Class C is a ref class, which calls class B with an explicit type :
ref class C {
public:
    C() { b = new B(); }
    ~C() {  }
    !C() { delete b; }
    void foo(int blah) { b->foo(blah); }
private:
    B* b;
};