我在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)
答案 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;
};