问题描述
|
我在CLI项目中传递模板时遇到了问题(我认为这是原因)。
我有3层类(出于封装原因和技术原因)。
我将展示一个有问题的部分的示例(请不要对封装原因做任何评论,这里不再展示)
类A在C ++ DLL中:
class A {
public:
template<class T>
void foo(T blah) { //Do stuff }
}
B类包装了A类(也是常规的非引用类):
class B {
public:
template<class T>
void foo(T blah) { a->foo(blah); }
private:
A* a;
}
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
&,int const&)\“
(?? $ foo @ _N @ B @ Namespace @@ $$ FQEAAXAEBVCustomString @ utils @@ AEB_N @ Z)
在函数中引用“私有:无效
__clrcall C :: foo(int)\“(?? $ foo @ _N @ Namespace @@ $$ FAE $ AAMXPE $ AAVString @ System @@ __ N @ Z)
错误LNK2019:未解决的外部
符号错误LNK2019:未解决
外部符号\“ public:void __cdecl
B :: foo(int int const&)\“
$ foo @ _N @ B @ Namespace @@ $$ FQEAAXAEBVCustomString @ utils @@ AEB_N @ Z)
在函数中引用“私有:无效
__clrcall C :: foo(int)\“(?? $ foo @ _N @ Namespace @@ $$ FAE $ AAMXPE $ AAVString @ System @@ __ N @ Z)
编辑
我现在没有线路(不在同一台PC上),但是它说它无法链接C.foo中引用的B.foo。
我在调试模式/ MDd中用/ clr编译ref类(是的,由于其他依赖项都以相同方式编译,因此它必须在调试模式下)
有人知道为什么会这样吗?更重要的是:该如何解决?
编辑:
将类B(包装程序)设置为使用/ GL(整个程序优化)进行编译时,出现了另一个错误:
LNK2001:
错误LNK2001:未解决
外部符号“ public:布尔型__cdecl
接口:: B :: foo(int&)const \“
(?? $ foo @ _J @ B @ Namespace @@ $$ FQEBA_NAEBVCustomString @ 123 @ AEA_J @ Z)
解决方法
问题是Visual Studio的链接器存在错误。
首先,您需要设置/ GL(整个程序优化)标志,以便它在编译后链接模板。
然后,您需要使用以下变通办法之一:
使用嵌套类模板时,在成员函数上的BUG:LNK2001
,这是C ++模板的标准损失,它们没有.NET泛型之类的外部链接。您必须将模板函数定义放在头文件中,以便可以将其#include包含在包含ref类的源代码文件中。就像您的代码片段一样。它不是特定于C ++ / CLI的。
如果您编译没有/ clr选项的其余本机类,请确保使用#pragma托管的方法将#include包裹起来(应该这样)。像这样:
#pragma managed(push,off)
#include \"FileWithTemplateDefinition.h\"
#pragma managed(pop)
,这些链接器错误指出您缺少某处的定义(而不是声明)。无法访问您的代码使我们难以确定,尤其是在没有任何冗长的链接器错误的情况下。
[编辑]
这段代码对我来说可以在托管的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;
};