학습
구조 마샬링 사용자 지정
구조체의 기본 마샬링 규칙이 필요한 것과 정확히 일치하지 않는 경우가 있습니다. .NET 런타임에서는 구조체 레이아웃과 필드 마샬링 방식을 사용자 지정할 수 있는 몇 가지 확장점을 제공합니다. 구조 레이아웃 사용자 지정은 모든 시나리오에서 지원되지만 필드 마샬링 사용자 지정은 런타임 마샬링이 사용하도록 설정된 시나리오에서만 지원됩니다. 런타임 마샬링이 사용하지 않도록 설정된 경우 모든 필드 마샬링을 수동으로 수행해야 합니다.
참고
이 문서에서는 원본 생성 interop을 위한 마샬링 사용자 지정을 다루지 않습니다. P/Invokes용 원본 생성 interop 또는 COM을 사용하는 경우 마샬링 사용자 지정을 참조하세요.
.NET에서는 메모리에 필드가 배치되는 방식을 사용자 지정할 수 있도록 System.Runtime.InteropServices.StructLayoutAttribute 특성과 System.Runtime.InteropServices.LayoutKind 열거형을 제공합니다. 다음 지침은 일반적인 문제를 방지하는 데 도움이 됩니다.
✔️ 가능한 경우 항상 LayoutKind.Sequential
을 사용하는 것이 좋습니다.
✔️ 네이티브 구조체에 공용 구조체 등의 명시적인 레이아웃도 있는 경우에만 마샬링할 때 LayoutKind.Explicit
을 사용하세요.
❌ 클래스를 사용하여 상속을 통해 복잡한 네이티브 형식을 표현하지 마세요.
❌ .NET Core 3.0보다 이전의 런타임을 대상으로 해야 하는 경우 비 Windows 플랫폼에서 구조체를 마샬링할 때 LayoutKind.Explicit
을 사용하지 마세요. .NET Core 3.0보다 이전의 런타임을 통해 Intel 또는 AMD 64비트 비 Windows 시스템에서 명시적 구조체를 네이티브 함수에 값으로 전달할 수 없습니다. 그러나 런타임은 모든 플랫폼에서 명시적 구조체를 참조로 전달하는 기능을 지원합니다.
네이티브 코드에는 여러 부울 표현이 있습니다. Windows만 해도 부울 값을 나타내는 세 가지 방법이 있습니다. 런타임은 구조체의 네이티브 정의를 알 수 없으므로, 부울 값을 나타내는 최상의 방법은 부울 값을 마샬링하는 방식을 추측하는 것입니다. .NET 런타임에서 부울 필드를 마샬링하는 방식을 나타내는 방법을 제공합니다. 다음 예에서는 .NET bool
을 여러 네이티브 부울 형식으로 마샬링하는 방법을 보여 줍니다.
다음 예제와 같이 부울 값은 기본적으로 네이티브 4바이트 Win32 BOOL
값으로 마샬링되도록 설정됩니다.
public struct WinBool
{
public bool b;
}
struct WinBool
{
public BOOL b;
};
명시적으로 설정하려는 경우 UnmanagedType.Bool 값을 사용하여 위와 동일한 동작을 얻을 수 있습니다.
public struct WinBool
{
[MarshalAs(UnmanagedType.Bool)]
public bool b;
}
struct WinBool
{
public BOOL b;
};
아래의 UnmanagedType.U1
또는 UnmanagedType.I1
값을 사용하여 b
필드를 1바이트 네이티브 bool
형식으로 마샬링하도록 런타임에 지정할 수 있습니다.
public struct CBool
{
[MarshalAs(UnmanagedType.U1)]
public bool b;
}
struct CBool
{
public bool b;
};
Windows에서는 UnmanagedType.VariantBool 값을 사용하여 부울 값을 2바이트 VARIANT_BOOL
값으로 마샬링하도록 런타임에 지정할 수 있습니다.
public struct VariantBool
{
[MarshalAs(UnmanagedType.VariantBool)]
public bool b;
}
struct VariantBool
{
public VARIANT_BOOL b;
};
참고
VARIANT_BOOL
은 VARIANT_TRUE = -1
및 VARIANT_FALSE = 0
인 점에서 대부분의 bool 형식과는 다릅니다. 또한 VARIANT_TRUE
와 같지 않은 모든 값은 false로 간주됩니다.
.NET에는 배열 마샬링을 사용자 지정하는 몇 가지 방법도 있습니다.
기본적으로 .NET에서는 배열을 인접한 요소 목록에 대한 포인터로 마샬링합니다.
public struct DefaultArray
{
public int[] values;
}
struct DefaultArray
{
int32_t* values;
};
COM API와 인터페이스하는 경우 배열을 SAFEARRAY*
개체로 마샬링해야 할 수도 있습니다. System.Runtime.InteropServices.MarshalAsAttribute 및 UnmanagedType.SafeArray 값을 사용하여 배열을 SAFEARRAY*
로 마샬링하도록 런타임에 지정할 수 있습니다.
public struct SafeArrayExample
{
[MarshalAs(UnmanagedType.SafeArray)]
public int[] values;
}
struct SafeArrayExample
{
SAFEARRAY* values;
};
SAFEARRAY
에 포함되는 요소 형식을 사용자 지정해야 하는 경우 MarshalAsAttribute.SafeArraySubType 및 MarshalAsAttribute.SafeArrayUserDefinedSubType 필드를 사용하여 SAFEARRAY
의 정확한 요소 형식을 사용자 지정할 수 있습니다.
현재 위치에서 배열을 마샬링해야 하는 경우 UnmanagedType.ByValArray 값을 사용하여 현재 위치에서 배열을 마샬링하도록 마샬러에 지정할 수 있습니다. 이 마샬링을 사용하는 경우, 런타임에서 구조체의 공간을 올바르게 할당할 수 있도록 배열의 요소 수에 대한 MarshalAsAttribute.SizeConst 필드에도 값을 제공해야 합니다.
public struct InPlaceArray
{
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 4)]
public int[] values;
}
struct InPlaceArray
{
int values[4];
};
참고
.NET에서는 가변 길이 배열 필드를 C99 유연한 배열 멤버로 마샬링할 수 없습니다.
.NET에서는 문자열 필드를 마샬링하기 위한 다양한 사용자 지정도 제공합니다.
기본적으로 .NET은 문자열을 Null 종료 문자열에 대한 포인터로 마샬링합니다. 인코딩은 System.Runtime.InteropServices.StructLayoutAttribute의 StructLayoutAttribute.CharSet 필드 값에 따라 다릅니다. 특성을 지정하지 않으면 인코딩은 기본적으로 ANSI 인코딩으로 설정됩니다.
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
public struct DefaultString
{
public string str;
}
struct DefaultString
{
char* str;
};
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
public struct DefaultString
{
public string str;
}
struct DefaultString
{
char16_t* str; // Could also be wchar_t* on Windows.
};
다른 필드에 다른 인코딩을 사용해야 하거나 구조체 정의를 좀 더 명시적으로 설정하려는 경우 System.Runtime.InteropServices.MarshalAsAttribute 특성에 UnmanagedType.LPStr 또는 UnmanagedType.LPWStr 값을 사용할 수 있습니다.
public struct AnsiString
{
[MarshalAs(UnmanagedType.LPStr)]
public string str;
}
struct AnsiString
{
char* str;
};
public struct UnicodeString
{
[MarshalAs(UnmanagedType.LPWStr)]
public string str;
}
struct UnicodeString
{
char16_t* str; // Could also be wchar_t* on Windows.
};
UTF-8 인코딩을 사용하여 문자열을 마샬링하려는 경우 MarshalAsAttribute의 UnmanagedType.LPUTF8Str 값을 사용할 수 있습니다.
public struct UTF8String
{
[MarshalAs(UnmanagedType.LPUTF8Str)]
public string str;
}
struct UTF8String
{
char* str;
};
참고
UnmanagedType.LPUTF8Str을 사용하려면 .NET Framework 4.7 이상 버전이나 .NET Core 1.1 이상 버전이 필요합니다. .NET Standard 2.0에서는 사용할 수 없습니다.
COM API로 작업하는 경우 문자열을 BSTR
로 마샬링해야 할 수도 있습니다. UnmanagedType.BStr 값을 사용하여 문자열을 BSTR
로 마샬링할 수 있습니다.
public struct BString
{
[MarshalAs(UnmanagedType.BStr)]
public string str;
}
struct BString
{
BSTR str;
};
WinRT 기반 API를 사용하는 경우 문자열을 HSTRING
로 마샬링해야 할 수도 있습니다. UnmanagedType.HString 값을 사용하여 문자열을 HSTRING
로 마샬링할 수 있습니다. HSTRING
마샬링은 WinRT 지원이 기본 제공되는 런타임에서만 지원됩니다. WinRT 지원은 .NET 5에서 제거되었으므로 .NET 5 이상에서는 HSTRING
마샬링이 지원되지 않습니다.
public struct HString
{
[MarshalAs(UnmanagedType.HString)]
public string str;
}
struct BString
{
HSTRING str;
};
API에서 구조체의 현재 위치에 문자열을 전달해야 하는 경우 UnmanagedType.ByValTStr 값을 사용할 수 있습니다. ByValTStr
에 의해 마샬링된 문자열의 인코딩은 CharSet
특성으로 결정됩니다. 또한 문자열 길이가 MarshalAsAttribute.SizeConst 필드에 의해 전달되어야 합니다.
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
public struct DefaultString
{
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 4)]
public string str;
}
struct DefaultString
{
char str[4];
};
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
public struct DefaultString
{
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 4)]
public string str;
}
struct DefaultString
{
char16_t str[4]; // Could also be wchar_t[4] on Windows.
};
Windows에서 작업하는 경우 네이티브 CY
또는 CURRENCY
구조체를 사용하는 일부 API를 발견할 수 있습니다. 기본적으로 .NET decimal
형식은 네이티브 DECIMAL
구조체로 마샬링됩니다. 그러나 UnmanagedType.Currency 값과 함께 MarshalAsAttribute를 사용하여 decimal
값을 네이티브 CY
값으로 변환하도록 마샬러에 지정할 수 있습니다.
public struct Currency
{
[MarshalAs(UnmanagedType.Currency)]
public decimal dec;
}
struct Currency
{
CY dec;
};
공용체는 동일한 메모리 위에 다양한 형식의 데이터를 포함할 수 있는 데이터 형식입니다. C 언어의 일반적인 데이터 형식입니다. 공용체는 LayoutKind.Explicit
을 사용하여 .NET에서 표현할 수 있습니다. .NET에서 공용체를 정의할 때 구조체를 사용하는 것이 좋습니다. 클래스를 사용하면 레이아웃 문제가 발생하고 예측할 수 없는 동작이 발생할 수 있습니다.
struct device1_config
{
void* a;
void* b;
void* c;
};
struct device2_config
{
int32_t a;
int32_t b;
};
struct config
{
int32_t type;
union
{
device1_config dev1;
device2_config dev2;
};
};
public unsafe struct Device1Config
{
void* a;
void* b;
void* c;
}
public struct Device2Config
{
int a;
int b;
}
public struct Config
{
public int Type;
public _Union Anonymous;
[StructLayout(LayoutKind.Explicit)]
public struct _Union
{
[FieldOffset(0)]
public Device1Config Dev1;
[FieldOffset(0)]
public Device2Config Dev2;
}
}
Windows에서 object
형식 필드를 네이티브 코드로 마샬링할 수 있습니다. 이러한 필드는 다음 세 가지 형식 중 하나로 마샬링할 수 있습니다.
기본적으로 object
형식 필드는 개체를 래핑하는 IUnknown*
으로 마샬링됩니다.
public struct ObjectDefault
{
public object obj;
}
struct ObjectDefault
{
IUnknown* obj;
};
개체 필드를 IDispatch*
로 마샬링하려는 경우 UnmanagedType.IDispatch 값과 함께 MarshalAsAttribute를 추가합니다.
public struct ObjectDispatch
{
[MarshalAs(UnmanagedType.IDispatch)]
public object obj;
}
struct ObjectDispatch
{
IDispatch* obj;
};
VARIANT
로 마샬링하려는 경우 UnmanagedType.Struct 값과 함께 MarshalAsAttribute를 추가합니다.
public struct ObjectVariant
{
[MarshalAs(UnmanagedType.Struct)]
public object obj;
}
struct ObjectVariant
{
VARIANT obj;
};
다음 표에서는 obj
필드의 여러 런타임 형식이 VARIANT
에 저장된 다양한 형식에 매핑되는 방식을 설명합니다.
.NET 형식 | VARIANT 형식 |
---|---|
byte |
VT_UI1 |
sbyte |
VT_I1 |
short |
VT_I2 |
ushort |
VT_UI2 |
int |
VT_I4 |
uint |
VT_UI4 |
long |
VT_I8 |
ulong |
VT_UI8 |
float |
VT_R4 |
double |
VT_R8 |
char |
VT_UI2 |
string |
VT_BSTR |
System.Runtime.InteropServices.BStrWrapper |
VT_BSTR |
object |
VT_DISPATCH |
System.Runtime.InteropServices.UnknownWrapper |
VT_UNKNOWN |
System.Runtime.InteropServices.DispatchWrapper |
VT_DISPATCH |
System.Reflection.Missing |
VT_ERROR |
(object)null |
VT_EMPTY |
bool |
VT_BOOL |
System.DateTime |
VT_DATE |
decimal |
VT_DECIMAL |
System.Runtime.InteropServices.CurrencyWrapper |
VT_CURRENCY |
System.DBNull |
VT_NULL |
.NET 피드백
.NET은(는) 오픈 소스 프로젝트입니다. 다음 링크를 선택하여 피드백을 제공해 주세요.