Импорт DLL и использование функции extern в C ++ - PullRequest
1 голос
/ 05 февраля 2012

Это, должно быть, основной вопрос, но я слишком долго борюсь с этой проблемой.Я искал повсюду в Google и нашел несколько похожих проблем и решений, но ни одно из них не решило мою конкретную проблему.

Я написал очень простую C ++ DLL.На самом деле это почти в стиле C, потому что в DLL есть только файл кода main.cpp с функцией, поэтому он даже не использует класс.

Тогда у меня есть два заголовочных файла:

MqlUtils.h:

#ifndef MQLUTILS_H
#define MQLUTILS_H

struct MqlStr
{
    int len;
    char *string;
};

enum TradeOperation
{
    OP_BUY = 0,
    OP_SELL = 1,
    OP_BUYLIMIT = 2,
    OP_SELLLIMIT = 3,
    OP_BUYSTOP = 4,
    OP_SELLSTOP = 5
};

#endif

main.h:

#ifndef _DLL_H_
#define _DLL_H_

#include "MqlUtils.h"

#define MT4_EXPFUNC __declspec(dllexport)

#define export extern "C" __declspec( dllexport )

 MT4_EXPFUNC int __stdcall  GetOrdersDetailsNoSymbol(const int orderCount, const char * MasterLicense, const char * SlaveLicense, int orderTicket[], int op[],
    double orderOpenPrice[], double orderStoploss[],
    double orderTakeProfit[], double orderLots[], int orderDateTime[],
    MqlStr * ordersymbol, MqlStr * ordercomments, int lotsCopyingMethod[], int returnedOrders[]);

#endif /* _DLL_H_ */

Фактически, для создания моей DLL, я начал с существующего кода, который кто-то написал, поэтому файл .cppу моей DLL есть какой-то неясный синтаксис, который я даже не уверен, что он делает.Вот выдержка из того, как выглядит .cpp:

#include "main.h"

#define _UNICODE 1
#define UNICODE 1
#define WIN32_LEAN_AND_MEAN

// ...

EXTERN_C IMAGE_DOS_HEADER __ImageBase;


#if BUILDING_DLL
#define DLLIMPORT __declspec (dllexport)
#else /* Not BUILDING_DLL */
#define DLLIMPORT __declspec (dllimport)
#endif /* Not BUILDING_DLL */

// ...

#ifdef __cplusplus
extern "C"
{
#endif

// ... function code

#ifdef __cplusplus
}
#endif

Я не включил все, что есть в файле .cpp, где есть // ..., есть что-то еще, но это базовые вещи, которые я понимаюну, так что это не должно быть источником моей проблемы ... Я буду рад опубликовать больше по мере необходимости.

Я не эксперт со всеми непонятными ключевыми словами, такими как __declspec и тому подобное, но DLL, как она есть, может быть успешно импортирована, и функция GetOrdersDetailsNoSymbol может использоваться какой-то программой, а именно MetaTrader 4 (которая являетсяосновная цель моей библиотеки).

Но теперь я хотел бы иметь возможность протестировать свою библиотеку с помощью программы на C ++, поэтому я создал пустую консольную программу, добавил проект библиотеки в ссылки на проект тестирования и связалФайлы .obj и .h через свойства проекта тестирования.

В настоящее время я получаю это при компиляции проекта тестирования:

Error   2   error LNK1120: 1 unresolved externals   Z:\Codes\Debug\TestsCpp.exe TestsCpp
Error   1   error LNK2019: unresolved external symbol "__declspec(dllimport) int __cdecl GetOrdersDetailsNoSymbol(int,char *,char *,int * const,int * const,double * const,double * const,double * const,double * const,int * const,struct MqlStr *,struct MqlStr *,int * const,int * const)" (__imp_?GetOrdersDetailsNoSymbol@@YAHHPAD0QAH1QAN2221PAUMqlStr@@311@Z) referenced in function "void __cdecl TestClient(void)" (?TestClient@@YAXXZ)    Z:\Codes\TestsCpp\main.obj  TestsCpp

О, и вот main.cpp для проекта тестирования:

#include "MqlUtils.h"
#include "main.h"

extern __declspec(dllimport) int GetOrdersDetailsNoSymbol(int orderCount, char * MasterLicense, char * SlaveLicense, int orderTicket[], int op[],
    double orderOpenPrice[], double orderStoploss[],
    double orderTakeProfit[], double orderLots[], int orderDateTime[],
    MqlStr* ordersymbol, MqlStr* ordercomments, int lotsCopyingMethod[], int returnedOrders[]);


void TestClient()
{
   char* Master = "7C83C4C2";
   char* Slave = "3B7C22A";

   int returnedOrderCount[1] = {0};

   double aStoredOrderOpenPrice[4];
   int aStoredOrderType[4];
   int aStoredOrderTicket[4];
   double aStoredOrderStopLoss[4];
   double aStoredOrdeTakeProfit[4];
   double aStoredOrderLots[4];
   int aStoredOrderDateTime[4];
   int aStoredLotsMethods[4];
   MqlStr* aStoredOrderComment[4];
   MqlStr* aStoredOrderSymbol[4];

   for (int i = 0; i < 4; i++)
   {
      aStoredOrderOpenPrice[i]= -1;
      aStoredOrderType[i]= -1;
      aStoredOrderTicket[i]= -1;
      aStoredOrderStopLoss[i]= -1;
      aStoredOrdeTakeProfit[i]= -1;
      aStoredOrderLots[i]= -1;
      aStoredOrderDateTime[i]= -1;
      aStoredLotsMethods[i]= -1;

      aStoredOrderComment[i]->len = 56;
      aStoredOrderComment[i]->string = "11111111111111111111111111111111111111111111111111111111";
      aStoredOrderSymbol[i]->len = 56;
      aStoredOrderSymbol[i]->string = "11111111111111111111111111111111111111111111111111111111";
   }


   GetOrdersDetailsNoSymbol(1, Master, Slave, aStoredOrderTicket,  aStoredOrderType,
                                            aStoredOrderOpenPrice, aStoredOrderStopLoss,
                                            aStoredOrdeTakeProfit, aStoredOrderLots, aStoredOrderDateTime,
                                            *aStoredOrderSymbol, *aStoredOrderComment,     aStoredLotsMethods, returnedOrderCount);
}

int main(int argc, char **argv)
{
   TestClient();

   return 0;
}

Если бы кто-нибудь мог помочь мне решить эту проблему, я был бы бесконечно благодарен.

Спасибо за чтение!

1 Ответ

3 голосов
/ 05 февраля 2012

Ваши два объявления GetOrdersDetailsNoSymbol не совпадают. В вашем заголовочном файле вы объявляете это с помощью __stdcall, а в main.cpp - нет. У вас должна быть только одна декларация. Он может использовать #define и #ifdef для применения ключевых слов dllimport или dllexport в зависимости от ситуации.

РЕДАКТИРОВАТЬ: Кроме того, избавиться от внешних "C" операторов. А затем используйте DLLIMPORT #define для объявления вашей функции и только #define BUILDING_DLL в сборке вашей DLL.

...