Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일

C# - 값 형식(Blittable)을 메모리 복사를 이용해 바이트 배열로 직렬화/역직렬화

아래의 동영상을 보면,

고성능 데이터 직렬화 라이브러리 MemoryPack 소개 with 최흥배
; https://youtu.be/D2DSXJHoQJo

흥미로운 코드가 나오는데요, 바로 값 형식(좀 더 정확한 제약으로는 ref struct를 만족할 수 있는 타입)에 대한 직렬화/역직렬화하는 코드입니다.

// .NET 5부터 사용 가능
public static byte[] SerializeBlittable<T>(in T? value)
{
    if (!RuntimeHelpers.IsReferenceOrContainsReferences<T>())
    {
        var buffer = GC.AllocateUninitializedArray<byte>(Unsafe.SizeOf<T>());
        Unsafe.WriteUnaligned(ref MemoryMarshal.GetArrayDataReference(buffer), value);
        return buffer;
    }

    return Array.Empty<byte>();
}

private static void DeserializeBlittable<T>(byte[] buffer, out T value)
{
    if (!RuntimeHelpers.IsReferenceOrContainsReferences<T>())
    {
        value = Unsafe.ReadUnaligned<T>(ref MemoryMarshal.GetArrayDataReference(buffer));
        return;
    }

    value = default!;
}

사용은 대충 이렇게 할 수 있습니다.

{
    var value = new TestStruct { A = 1, B = 2 };
    var bytes = SerializeBlittable(value);
    Console.WriteLine(BitConverter.ToString(bytes)); // 출력 결과: 01-00-00-00-02-00-00-00

    value = new TestStruct { A = 0, B = 0 };
    DeserializeBlittable(bytes, out value);
    Console.WriteLine(value); // 출력 결과: A: 1, B: 2
}

internal struct TestStruct
{
    public int A { get; set; }
    public int B { get; set; }

    public override string ToString() => $"A: {A}, B: {B}";
}

별다른 직렬화 코드 없이 메모리 상태를 그대로 복사하는 것이기 때문에 고속일 수밖에 없습니다. MemoryPack 패키지가 괜히 빠른 게 아니겠지요. ^^

단지, 메모리 직렬화이기 때문에 Endian이 CPU를 따라간다는 점만 유의하시면 되겠습니다.




참고로, 위의 코드에서 AllocateUninitializedArray 메서드가 .NET 5부터 지원되고 있어 최소 런타임 기준으로 닷넷 5라고 주석을 단 것인데요, 사실 이런 코드를 예전에도 unsafe 구문과 섞어 사용할 수는 있었습니다.

// .NET Framework에서도 가능

public static byte[] SerializePtr<T>(T* pValue) where T : unmanaged
{
    int size = Marshal.SizeOf(*pValue);
    byte[] buffer = new byte[size];

    fixed (byte* pBuffer = buffer)
    {
        Buffer.MemoryCopy(pValue, pBuffer, size, size);
    }

    return buffer;
}

public static void DeserializePtr<T>(byte[] buffer, out T value) where T : unmanaged
{
    int size = Marshal.SizeOf<T>();
    fixed(T* pValue = &value)
    fixed (byte* pBuffer = buffer)
    {
        Buffer.MemoryCopy(pBuffer, pValue, size, size);
    }
}

사용 방법도 역시 별반 다르지 않습니다.

{
    var value = new TestStruct { A = 1, B = 2 };
    var bytes = SerializePtr(&value);
    Console.WriteLine(BitConverter.ToString(bytes)); // 출력 결과: 01-00-00-00-02-00-00-00

    value = new TestStruct { A = 0, B = 0 };
    DeserializePtr(bytes, out value);
    Console.WriteLine(value); // 출력 결과: A: 1, B: 2
}

가만 보면, 요즘 닷넷 런타임 변화의 트렌드 중 하나가 unsafe 예약어 사용이 필요 없는 방향으로 움직이는 듯합니다. 기존의 unsafe 사용은 잘 사용한다고 해도 어쨌든 managed 환경의 제어를 받지 못하므로, 최대한 관리 영역으로 끌어내 프로그램의 동작을 안정적으로 유지하려는 노력이 나름 기특한 것 같습니다. ^^

(첨부 파일은 이 글의 예제 코드를 포함합니다.)





[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]







[최초 등록일: ]
[최종 수정일: 12/5/2023]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... [61]  62  63  64  65  66  67  68  69  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
12530정성태2/4/202124678개발 환경 구성: 536. Wireshark + C#으로 확인하는 TCP 통신의 Receive Window
12529정성태2/4/202124112개발 환경 구성: 535. Wireshark + C#으로 확인하는 TCP 통신의 MIN RTO [1]
12528정성태2/1/202124607개발 환경 구성: 534. Wireshark + C#으로 확인하는 TCP 통신의 MSS(Maximum Segment Size) - 윈도우 환경
12527정성태2/1/202124387개발 환경 구성: 533. Wireshark + C#으로 확인하는 TCP 통신의 MSS(Maximum Segment Size) - 리눅스 환경파일 다운로드1
12526정성태2/1/202121223개발 환경 구성: 532. Azure Devops의 파이프라인 빌드 시 snk 파일 다루는 방법 - Secure file
12525정성태2/1/202119734개발 환경 구성: 531. Azure Devops - 파이프라인 실행 시 빌드 이벤트를 생략하는 방법
12524정성태1/31/202119410개발 환경 구성: 530. 기존 github 프로젝트를 Azure Devops의 빌드 Pipeline에 연결하는 방법 [1]
12523정성태1/31/202122430개발 환경 구성: 529. 기존 github 프로젝트를 Azure Devops의 Board에 연결하는 방법
12522정성태1/31/202124974개발 환경 구성: 528. 오라클 클라우드의 리눅스 VM - 9000 MTU Jumbo Frame 테스트
12521정성태1/31/202122094개발 환경 구성: 527. 이더넷(Ethernet) 환경의 TCP 통신에서 MSS(Maximum Segment Size) 확인 [1]
12520정성태1/30/202121068개발 환경 구성: 526. 오라클 클라우드의 VM에 ping ICMP 여는 방법
12519정성태1/30/202120776개발 환경 구성: 525. 오라클 클라우드의 VM을 외부에서 접근하기 위해 포트 여는 방법
12518정성태1/30/202139195Linux: 37. Ubuntu에 Wireshark 설치 [2]
12517정성태1/30/202125717Linux: 36. 윈도우 클라이언트에서 X2Go를 이용한 원격 리눅스의 GUI 접속 - 우분투 20.04
12516정성태1/29/202122625Windows: 188. Windows - TCP default template 설정 방법
12515정성태1/28/202124712웹: 41. Microsoft Edge - localhost에 대해 http 접근 시 무조건 https로 바뀌는 문제 [3]
12514정성태1/28/202125069.NET Framework: 1021. C# - 일렉트론 닷넷(Electron.NET) 소개 [1]파일 다운로드1
12513정성태1/28/202120244오류 유형: 698. electronize - User Profile 디렉터리에 공백 문자가 있는 경우 빌드가 실패하는 문제 [1]
12512정성태1/28/202123683오류 유형: 697. The program can't start because VCRUNTIME140.dll is missing from your computer. Try reinstalling the program to fix this problem.
12511정성태1/27/202122705Windows: 187. Windows - 도스 시절의 8.3 경로를 알아내는 방법
12510정성태1/27/202123262.NET Framework: 1020. .NET Core Kestrel 호스팅 - Razor 지원 추가 [1]파일 다운로드1
12509정성태1/27/202121132개발 환경 구성: 524. Jupyter Notebook에서 C#(F#, PowerShell) 언어 사용을 위한 환경 구성 [3]
12508정성태1/27/202122722개발 환경 구성: 523. Jupyter Notebook - Slide 플레이 버튼이 없는 경우
12507정성태1/26/202122804VS.NET IDE: 157. Visual Studio - Syntax Visualizer 메뉴가 없는 경우
12506정성태1/25/202126279.NET Framework: 1019. Microsoft.Tye 기본 사용법 소개 [1]
12505정성태1/23/202120800.NET Framework: 1018. .NET Core Kestrel 호스팅 - Web API 추가 [1]파일 다운로드1
... [61]  62  63  64  65  66  67  68  69  70  71  72  73  74  75  ...