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

问题描述

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

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...