Вызов DLL-функции C ++ из C#:Из структур, строк и массивов wchar_t

StackOverflow https://stackoverflow.com/questions/680066

Вопрос

Вот простая проблема, которую мне нужно решить, но это заставляет меня чувствовать, что мои волосы седеют, поскольку все мои попытки возвращают мне одну и ту же ошибку:

"Предпринята попытка чтения или записи защищенной памяти.Это часто указывает на то, что другая память повреждена ".

У меня есть пример приложения, написанного на C ++, которое вызывает dll.Вот соответствующий код:

    //function I need to call
bool convertHKID_Name(char *code,RECO_DATA *o_data);    //hkid 

//struct definition
struct RECO_DATA{
    wchar_t FirstName[200];
    wchar_t Surname[200];
};

//how it is used in C++ code
CString code;
RECO_DATA data;
GetDlgItemText(IDC_CODE,code);
char _code[200];
WideCharToMultiByte(CP_UTF8, 0, code, -1, (char *)_code, 200, NULL, NULL);
ocr->convertHKID_Name(_code,&data)

Теперь, когда я отлаживаю код C ++, он делает все правильно - записывает некоторые данные в юникоде в структуру данных.

Вот моя попытка сделать то же самое на C#

    //my C# wrapper class
public class cnOCRsdk
{
    [StructLayout(LayoutKind.Sequential, Size=400, CharSet=CharSet.Unicode), Serializable]
    public struct RECO_DATA
    {
        [MarshalAsAttribute(UnmanagedType.ByValTStr, SizeConst = 200)]
        public string FirstName;
        [MarshalAsAttribute(UnmanagedType.ByValTStr, SizeConst = 200)]
        public string Surname;
    };

    [DllImport(@"cnOCRsdk.dll", CharSet=CharSet.Auto, EntryPoint = "?convertHKID_Name@CcnOCRsdk@@QAE_NPADPAURECO_DATA@@@Z")]
    public static extern bool convertHKID_Name(ref string num, ref RECO_DATA o_data);

    [DllImport("Kernel32.dll")]
    public static extern int WideCharToMultiByte(uint CodePage, uint dwFlags,
        [In, MarshalAs(UnmanagedType.LPWStr)]string lpWideCharStr,
        int cchWideChar,
        [Out, MarshalAs(UnmanagedType.LPStr)]StringBuilder lpMultiByteStr,
        int cbMultiByte,
        IntPtr lpDefaultChar, // Defined as IntPtr because in most cases is better to pass
        IntPtr lpUsedDefaultChar // NULL
        );
}

//my attempt to call the function from the dll
cnOCRsdk.RECO_DATA recoData = new cnOCRsdk.RECO_DATA();
string num = "262125355174";
StringBuilder sb = new StringBuilder(200, 200);
cnOCRsdk.WideCharToMultiByte(65001, 0, num, -1, sb, 200, IntPtr.Zero, IntPtr.Zero);
string sbTostring = sb.ToString();
//the next line generates the 'Attempted to read or write protected memory' error
bool res = cnOCRsdk.convertHKID_Name(ref sbTostring, out recoData);

Я предполагаю, что я неправильно маршалирую структуру RECO_DATA, потому что именно в эту структуру записывается функция convertHKID_Name .Но как я должен это исправить?

Это было полезно?

Решение

Я верю, что это должно сработать, если вы

  1. Измените декларацию на convertHKID_Name Для CharSet.Ansi
  2. Удалите "ref" из строки параметр
  3. Передайте string num непосредственно к convertHKID_Name вместо того, чтобы позвонить WideCharToMultiByte

Другие советы

Я написал управляемую оболочку на C ++ для моей неуправляемой библиотеки dll, но снова немного застрял.

Продолжение здесь

Передача параметров типа данных C # в dll, написанную на C ++?

Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top