Как связать DLL с моим основным проектом? (Получение неразрешенной внешней ошибки)

Я все еще изучаю некоторые способы использования языка C ++.

Итак, я решил создать свою библиотеку (динамическую) и импортировать ее в свой проект. Я выполнил некоторые шаги из учебника в Интернете, но у меня всегда есть ошибка неразрешенного внешнего ...

Разрешите перейти к проекту DLL:

File1.cpp:

#include "MathFuncsDll.h"
#include <stdexcept>

using namespace std;

namespace MathFuncs
{
    double MyMathFuncs::Add(double a, double b)
    {
        return a + b;
    }

    double MyMathFuncs::Subtract(double a, double b)
    {
        return a - b;
    }

    double MyMathFuncs::Multiply(double a, double b)
    {
        return a * b;
    }

    double MyMathFuncs::Divide(double a, double b)
    {
        if (b == 0)
        {
            throw invalid_argument("b cannot be zero!");
        }

        return a / b;
    }
}

MathFuncs.h:

#ifdef MATHFUNCSDLL_EXPORTS
#define MATHFUNCSDLL_API __declspec(dllexport)
#else
#define MATHFUNCSDLL_API __declspec(dllimport)
#endif

namespace MathFuncs
{
    // This class is exported from the MathFuncsDll.dll
    class MyMathFuncs
    {
    public:
        // Returns a + b
        static MATHFUNCSDLL_API double Add(double a, double b);

        // Returns a - b
        static MATHFUNCSDLL_API double Subtract(double a, double b);

        // Returns a * b
        static MATHFUNCSDLL_API double Multiply(double a, double b);

        // Returns a / b
        // Throws const std::invalid_argument& if b is 0
        static MATHFUNCSDLL_API double Divide(double a, double b);
    };
}

Результат: скомпилировано успешно (есть файлы Project1.dll и Project1.lib).

Запущено новое консольное приложение со следующими данными:

File1.cpp:

// MyExecRefsDll.cpp
// compile with: /EHsc /link MathFuncsDll.lib

#include <iostream>
#include <Windows.h>

#include "MathFuncsDll.h"

using namespace std;

int main()
{
    double a = 7.4;
    int b = 99;

    try {

        LoadLibrary(TEXT("MathFuncsDll.dll")); // Also tried without TEXT();

        cout << "a + b = " <<
            MathFuncs::MyMathFuncs::Add(a, b) << endl;
        cout << "a - b = " <<
            MathFuncs::MyMathFuncs::Subtract(a, b) << endl;
        cout << "a * b = " <<
            MathFuncs::MyMathFuncs::Multiply(a, b) << endl;
        cout << "a / b = " <<
            MathFuncs::MyMathFuncs::Divide(a, b) << endl;

        try
        {
            cout << "a / 0 = " <<
                MathFuncs::MyMathFuncs::Divide(a, 0) << endl;
        }
        catch (const invalid_argument &e)
        {
            cout << "Caught exception: " << e.what() << endl;
        }
    }
    catch (...){
            cout << "Problem when loading dll file" << endl;
    }

    system("pause");

    return 0;
}

PS.:

Я тоже пробовал без функции LoadLibrary().

Что-то, что я тоже пробовал: -> Добавил в проект файлы .lib, .h, .dll;

-> Добавлены файлы .lib, .h, .dll в ту же папку, что и в папке консольного приложения;

-> Добавлены файлы .lib, .h, .dll в ссылки проекта (общие параметры C ++).

Что я думаю: MathFuncsDLL.h читается компилятором, как только он находит функции / классы, когда я пишу код основной программы.

Проблемы, которые у меня остались до сих пор:

[Ошибка ilink32] Ошибка: неразрешенный внешний 'MathFuncs :: MyMathFuncs :: Add (double, double)', на который имеется ссылка в C: \ USERS \ MAURO \ DESKTOP \ PROJETO \ WIN32 \ DEBUG \ FILE1.OBJ

[Ошибка ilink32] Ошибка: неразрешенный внешний объект «MathFuncs :: MyMathFuncs :: Subtract (double, double)», указанный в C: \ USERS \ MAURO \ DESKTOP \ PROJETO \ WIN32 \ DEBUG \ FILE1.OBJ

[Ошибка ilink32] Ошибка: неразрешенный внешний объект «MathFuncs :: MyMathFuncs :: Multiply (double, double)», указанный в C: \ USERS \ MAURO \ DESKTOP \ PROJETO \ WIN32 \ DEBUG \ FILE1.OBJ

[Ошибка ilink32] Ошибка: неразрешенный внешний 'MathFuncs :: MyMathFuncs :: Divide (double, double)', на который имеется ссылка в C: \ USERS \ MAURO \ DESKTOP \ PROJETO \ WIN32 \ DEBUG \ FILE1.OBJ

Подробная информация о компиляторе: -> C ++ Builder XE7.

С этого момента большое спасибо.


person mauroaraujo    schedule 29.10.2015    source источник


Ответы (1)


Вы используете LoadLibrary() неправильно и бесполезно. Вы не передаете возвращаемый дескриптор модуля в GetProcAddress() для динамической загрузки функций DLL. Так что удалите вызов LoadLibrary().

Код вашей консоли пытается статически связываться с функциями DLL. Чтобы разрешить ссылки, вам необходимо добавить .lib файл DLL в свой консольный проект либо в Диспетчере проектов, либо с помощью оператора #pragma comment(lib, Project1.lib) в вашем коде. Недостаточно наличия файла .lib в папке проекта консоли.

При этом ваша DLL не должна пытаться экспортировать статические функции класса с пространством имен для начала. Вместо этого экспортируйте плоские автономные функции в стиле C. Ваш заголовок может предоставить класс-оболочку с пространством имен для использования в C ++, только не экспортируйте его.

Также небезопасно выбрасывать исключения (особенно исключения на основе классов) через границу DLL. Вам нужно полностью избавиться от этого. В случае Divide(), либо заставьте вызывающего абонента подтвердить, что он никогда не проходит b=0, либо измените подпись Divide(), чтобы вернуть bool, указывающий на успех / неудачу, и используйте отдельный выходной параметр для возврата результата деления.

Попробуйте что-нибудь еще вроде этого:

MathFuncsDll.cpp:

#define MATHFUNCSDLL_EXPORTS
#include "MathFuncsDll.h"

double MathFuncs_Add(double a, double b)
{
    return a + b;
}

double MathFuncs_Subtract(double a, double b)
{
    return a - b;
}

double MathFuncs_Multiply(double a, double b)
{
    return a * b;
}

double MathFuncs_Divide(double a, double b)
{
    return a / b;
}
/* alternatively:
bool MathFuncs_Divide(double a, double b, double *result)
{
    if (b == 0) return false;
    if (result) *result = a / b;
    return true;
}
*/

MathFuncsDll.h:

#ifndef MathFuncsDllH
#define MathFuncsDllH

#ifdef MATHFUNCSDLL_EXPORTS
#define MATHFUNCSDLL_API __declspec(dllexport)
#else
#define MATHFUNCSDLL_API __declspec(dllimport)
#endif

#ifdef __cplusplus
extern "C" {
#endif

// Returns a + b
MATHFUNCSDLL_API double MathFuncs_Add(double a, double b);

// Returns a - b
MATHFUNCSDLL_API double MathFuncs_Subtract(double a, double b);

// Returns a * b
MATHFUNCSDLL_API double MathFuncs_Multiply(double a, double b);

// Returns a / b
MATHFUNCSDLL_API double MathFuncs_Divide(double a, double b);
// alternatively: bool MathFuncs_Divide(double a, double b, double *result);

#ifdef __cplusplus
}

#include <stdexcept>

namespace MathFuncs
{
    class MyMathFuncs
    {
    public:
        static Add(double a, double b) { return MathFuncs_Add(a, b); }
        static double Subtract(double a, double b) { return MathFuncs_Subtract(a, b); }
        static double Multiply(double a, double b) { return MathFuncs_Multiply(a, b); }
        static double Divide(double a, double b)
        {
            if (b == 0)
                throw std::invalid_argument("b cannot be zero!");
            return MathFuncs_Divide(a, b);
            /* alternatively:
            double result;
            if (!MathFuncs_Divide(a, b, &result))
                throw std::invalid_argument("b cannot be zero!");
            return result;
            */
        }
    };
}
#endif

#endif

File1.cpp:

#include <windows.h>
#include <iostream>

// if you don't add the DLL .lib file to the project using the Project Manager,
// uncomment this statement ...  either way, you really should rename your DLL
// project to something more meaningful then "Project1" ...
// #pragma comment(lib, "Project1.lib")

#include "MathFuncsDll.h"

int main()
{
    double a = 7.4;
    int b = 99;

    try
    {
        std::cout << "a + b = " << MathFuncs::MyMathFuncs::Add(a, b) << std::endl;
        std::cout << "a - b = " << MathFuncs::MyMathFuncs::Subtract(a, b) << std::endl;
        std::cout << "a * b = " << MathFuncs::MyMathFuncs::Multiply(a, b) << std::endl;
        std::cout << "a / b = " << MathFuncs::MyMathFuncs::Divide(a, b) << std::endl;

        try
        {
            std::cout << "a / 0 = " << MathFuncs::MyMathFuncs::Divide(a, 0) << std::endl;
        }
        catch (const std::invalid_argument &e)
        {
            std::cout << "Caught exception: " << e.what() << std::endl;
        }
    }
    catch (...)
    {
        std::cout << "Problem when loading dll file" << std::endl;
    }

    system("pause");

    return 0;
}
person Remy Lebeau    schedule 29.10.2015