Выделение простой строки в C # - PullRequest
2 голосов
/ 23 ноября 2010

Я полностью провалил этот действительно простой P / Invoke.

    [System.Runtime.InteropServices.DllImport(
        "temp.dll"
    )]
    private static extern System.IntPtr parser_alloc();

    [System.Runtime.InteropServices.DllImport(
        "temp.dll", 
        CharSet = System.Runtime.InteropServices.CharSet.Ansi
    )]
    private static extern void parser_add_file(
        System.IntPtr p,
        [MarshalAs(UnmanagedType.LPStr)]string f,
        [MarshalAs(UnmanagedType.LPStr)]string s);

    [System.Runtime.InteropServices.DllImport(
        "temp.dll"
    )]
    private static extern void parser_free(
        System.IntPtr p);

    [System.Runtime.InteropServices.DllImport(
        "temp.dll"
    )]
    private static extern void parser_emit_results(
        System.IntPtr p);

    [System.Runtime.InteropServices.DllImport(
        "temp.dll",
        CharSet = System.Runtime.InteropServices.CharSet.Ansi
    )]
    private static extern void parser_file_updated(
        System.IntPtr p,
        [MarshalAs(UnmanagedType.LPStr)]string f,
        int offset,
        int added);

Кажется, что создание объекта прошло хорошо.Однако, когда я пытаюсь вызвать parser_add_file, среда выполнения сообщает мне, что у меня несоответствующая подпись.Это мой существующий код C ++:

class Parser {
public:
    void add_file(std::string filename, std::string source) {
        std::cout << "add_file | " << filename << " | " << source << "\n";
    }
    void clear_contents() {
        std::cout << "clear_contents\n";
    }
    void emit_results() {
        std::cout << "emit_results\n";
    }
    void file_updated(std::string filename, int offset, int added) {
        std::cout << "file_updated | " << filename << " | " << offset << " | " << added << "\n";
        // added should be negative to signify removing
    }
    Parser() {
        std::cout << "I made a Parser!\n";
    }
};

extern "C" __declspec(dllexport) Parser* parser_alloc() {
    return new Parser;
};

extern "C" __declspec(dllexport) void parser_add_file(Parser* p, const char* filename, const char* string) {
    p->add_file(filename, string);
}

extern "C" __declspec(dllexport) void parser_free(Parser* p) {
    delete p;
}

 extern "C" __declspec(dllexport) void parser_emit_results(Parser* p) {
    p->emit_results();
}

 extern "C" __declspec(dllexport) void parser_file_updated(Parser* p, const char* filename, int offset, int added) {
    p->file_updated(filename, offset, added);
}

Должен ли я просто пропустить P / Invoke и перейти к C ++ / CLI для этой работы?

Ответы [ 2 ]

1 голос
/ 23 ноября 2010

Оказывается, мне пришлось вручную установить соглашение о вызовах cdecl - по умолчанию это stdcall.

0 голосов
/ 23 ноября 2010

Если у вас есть доступ к исходному тексту на c ++ (что выглядит так, как у вас), вы также можете изменить соглашение о вызовах на уровне декларации c ++.Вот пример функции для библиотеки, которую я написал:

int _stdcall GeoConvertLatLongToUTM(double latitude, double longitude, char *szOutput)
...