EntryPointNotFoundException при привязке dll C++ в C#

Я пытаюсь связать простую dll c++, показанную вhttp://msdn.microsoft.com/en-us/library/ms235636.aspx в моем консольном приложении c#, но я получаю EntryPointNotFoundException для добавления в dll во время выполнения. Мой тестовый класс

namespace BindingCppDllExample
{
    public class BindingDllClass
    {
        [DllImport("MathFuncsDll.dll")]
        public static extern double Add(double a, double b);
    }

    public class Program
    {
        public static void Main(string[] args)
        {
            double a = 2.3;
            double b = 3.8;
            double c = BindingDllClass.Add(a, b);

            Console.WriteLine(string.Format("{0} + {1} = {2}", a, b, c));
        }
    }
}

что неправильно?

2 ответов


вы можете попробовать объявить функции вне класса, а также экспортировать их с помощью extern "C":

:
// MathFuncsDll.h
namespace MathFuncs
{
    // Returns a + b
    extern "C" __declspec(dllexport) double Add(double a, double b);

    // Returns a - b
    extern "C" __declspec(dllexport) double Subtract(double a, double b);

    // Returns a * b
    extern "C" __declspec(dllexport) double Multiply(double a, double b);

    // Returns a / b
    // Throws DivideByZeroException if b is 0
    extern "C" __declspec(dllexport) double Divide(double a, double b);
}

реализация:

// MyMathFuncs.cpp
#include "MathFuncsDll.h"
#include <stdexcept>

using namespace std;

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

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

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

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

        return a / b;
    }
}

телефонный код:

namespace BindingCppDllExample
{
    public class BindingDllClass
    {
        [DllImport("MathFuncsDll.dll")]
        public static extern double Add(double a, double b);
    }

    public class Program
    {
        public static void Main(string[] args)
        {
            double a = 2.3;
            double b = 3.8;
            double c = BindingDllClass.Add(a, b);

            Console.WriteLine(string.Format("{0} + {1} = {2}", a, b, c));
        }
    }
}

в таких случаях, как этот, вы можете скачать Зависимость Walker, загрузите в него свою DLL и посмотрите на список функций экспорта. Вы также можете использовать команду dumpbin для этого.

по умолчанию функция, экспортированная из C++ или C DLL, использует Название Украшения (также называемый Имя Коверкая).

как заявил MSDN:

оформленное имя для функции C++ содержит следующее информация:

  • имя функции.
  • класс, членом которого является функция, если она является функцией-членом. Это может включать в себя класс, который заключает в себе класс функции, и так далее.
  • пространство имен, к которому принадлежит функция (если она является частью пространства имен).
  • типы параметров функции.
  • соглашение о вызовах.
  • возвращаемый тип функции.

так оформлен имя Add функция, например, будет выглядеть как Add@MyMathFuncs@MathFuncs@@SANNN@Z.

но можно заставить компилятор C++ выставить недекорированные имена для функций c++, заключив функцию и любые прототипы функций в extern "C" {…} блок, как предложил Дарин Димитров.

хотя, если вы собираетесь использовать стороннюю DLL (так что вы не можете изменить его) или вы просто не хотите выставлять украшенные имена по какой-то причине, вы можете Укажите имя функции явно:

[DllImport("MathFuncsDll.dll", EntryPoint = "Add@MyMathFuncs@MathFuncs@@SANNN@Z")]
public static extern double Add(double a, double b);