C#과 C DLL 간 문자열 전달: 최소 예
문제 설명
에도 불구하고 겉으로는 간단해 보이는 접근 방식에 따라 interop을 통해 C#과 C DLL 간에 문자열을 전달하면 System.AccessViolationException. 두 언어의 코드가 모두 올바른 것으로 나타나 문제의 근본 원인에 대한 혼란을 야기합니다.
C 코드
C 코드는 concat 함수를 정의합니다. 두 개의 문자열을 입력하고 결과를 반환합니다. 이 함수는 __declspec(dllexport)를 사용하여 DLL로 내보내도록 표시되어 있습니다.
extern "C" { string concat(string a, string b){ return a + b; } }
C# 코드
C#에서 코드는 DllImport를 사용하여 concat 함수를 가져옵니다. CallingConvention은 C 규칙과 일치하도록 Cdecl로 설정됩니다.
[DllImport("*****.dll", CallingConvention = CallingConvention.Cdecl)] static extern string concat(string a, string b);
예외 및 분석
C#에서 concat("a", "b")를 호출하면 오류가 발생합니다. System.AccessViolationException. 이 예외는 네이티브 코드 내에서 잘못된 메모리 액세스가 발생하고 있음을 나타냅니다.
해결책: Interop-Compatible Types
문제의 원인은 C std 사용에 있습니다. ::interop 경계를 넘는 문자열입니다. C#에서는 std::string 개체를 직접 만들거나 처리할 수 없습니다. 대신, 데이터 교환을 위해 interop 호환 유형을 사용해야 합니다.
Interop 친화적인 데이터 유형
적합한 interop 유형에는 문자열을 전달하기 위한 null로 끝나는 문자 배열이 포함됩니다. C#에서 C로 . C에서 C#으로 문자열을 전달할 때 호출자가 버퍼를 할당하고 호출 수신자가 버퍼에 쓰는 방식으로 StringBuilder를 사용할 수 있습니다.
수정된 코드
C에서, concat 함수는 char* 및 int 매개변수를 사용하여 선언되어 버퍼와 그 내용을 수신할 수 있습니다.
extern "C" { void concat(char *str, int len) { // Concatenate strings and write to buffer } }
C#에서는 StringBuilder를 활용할 수 있습니다.
[DllImport("...", CallingConvention = CallingConvention.Cdecl)] static extern void concat(StringBuilder str, int len); .... StringBuilder sb = new StringBuilder(10); concat(sb, sb.Capacity);
interop 호환 문자열 유형을 사용하면 C#과 C DLL 간의 문자열 교환을 용이하게 할 수 있습니다. 효과적으로.
위 내용은 C#과 C DLL 간에 문자열을 전달할 때 System.AccessViolationException을 방지하는 방법은 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!