1

我创建了一个 DLL 项目并成功构建了它。然后我尝试在另一个项目 TEST 中使用 DLL,但出现以下错误。

Error   1   error LNK2001: unresolved external symbol "public: void __thiscall SnoMessage::setRawMessageName(class ATL::CStringT<wchar_t,class StrTraitMFC_DLL<wchar_t,class ATL::ChTraitsCRT<wchar_t> > >)" (?setRawMessageName@SnoMessage@@QAEXV?$CStringT@_WV?$StrTraitMFC_DLL@_WV?$ChTraitsCRT@_W@ATL@@@@@ATL@@@Z)

我在链接器属性中添加了所需的库,并且还在 TEST 包含目录中添加了头文件。因此该功能正在被识别,但它不断给出这些错误。DLL 由以下文件组成

SnoMessage.h

#pragma once
#include "StdAfx.h"
class SnoMessage
{
public:
    __declspec(dllexport) SnoMessage(void);
    __declspec(dllexport) ~SnoMessage(void);
    __declspec(dllexport) void setRawMessageName(CString messageName);
    __declspec(dllexport) void setRawMessageType(CString messageType);
    __declspec(dllexport) void setRawMessageAttributes(std::map<CString,CString> attributes);
    __declspec(dllexport) CString getRawMessageName();
    __declspec(dllexport) CString getRawMessageType();
    __declspec(dllexport) std::map<CString,CString> getRawMessageAttributes();

private:
    CString messageName;
    CString messageType;
    std::map<CString,CString> attributes;
};

SnoMessage.cpp

#include "stdafx.h"
#include "SnoMessage.h"


SnoMessage::SnoMessage(void)
{
}


SnoMessage::~SnoMessage(void)
{
}

void SnoMessage::setRawMessageName(CString messageName){
    this->messageName = messageName;
}

void SnoMessage::setRawMessageType(CString messageType){
    this->messageType = messageType;
}

void SnoMessage::setRawMessageAttributes(std::map<CString,CString> attributes){
    this->attributes = attributes;
}

CString SnoMessage::getRawMessageName(){
    return messageName;
}

CString SnoMessage::getRawMessageType(){
    return messageType;
}

std::map<CString,CString> SnoMessage::getRawMessageAttributes(){
    return attributes;
}

在测试中,我正在执行以下操作:

测试.cpp

// test.cpp : Defines the entry point for the console application.
//

#include "stdafx.h"
#include "SnoMessage.h"

int _tmain(int argc, _TCHAR* argv[])
{
    SnoMessage *msg = new SnoMessage();
    msg->setRawMessageName("TEST");
    return 0;
}

如果您需要更多信息,请告诉我,谢谢。

4

4 回答 4

12

在您的 dll 中,在您要用于导出定义的某些标头中定义它...

我的出口

#ifdef SNOMESSAGE_EXPORTS
#define SNOMESSAGE_API __declspec(dllexport)
#else
#define SNOMESSAGE_API __declspec(dllimport)
#endif

现在在您的 dll 中,您只需定义 SNOMESSAGE_EXPORTS,然后当您的 dll 被编译时,您的类和方法将对 exe 可见。但是,当您在 exe 中包含这些相同的标头时,宏将导入它们而不是导出。

 //In the DLL this is == to export, in the executable this is import.  Problem solved.
class SNOMESSAGE_API SnoMessage
{
public:
//...
};

您不再需要导出每个成员,只需导出类。

于 2012-04-10T21:57:29.670 回答
0

我会将整个类标记为已导出,而不仅仅是其成员函数。此外,按照此对话的建议,您需要指定__declspec(dllecport)__declspec(dllimport)基于您是在 DLL 中包含标头还是使用 DLL 的代码;并在 DLL 项目中定义保护宏。

于 2012-04-09T21:28:59.697 回答
0

编译 DLL 时应该有 __declspec(dllexport),但编译 exe 时应该有 __declspec(dllimport)。执行此操作的最简单方法是在“在 DLL 中”和“在 DLL 之外”时具有不同值的 #define 某处。还要导出整个类而不是单个方法。

于 2012-04-09T22:06:12.320 回答
0

有一种情况,dll编译使用C调用,而exe使用标准调用,x64下的链接没有问题,但是使用win32的时候会显示这个链接错误2001。这种情况只需要对win32平台的dll和exe都使用C调用(https://docs.microsoft.com/en-us/cpp/error-messages/tool-errors/name-decoration?view=msvc-160)。

于 2021-01-20T00:48:41.537 回答