Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (seongtaejeong at gmail.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)

C# - Collection 식을 지원하는 사용자 정의 타입을 CollectionBuilder 특성으로 성능 보완

컬렉션 식이 C# 12에 추가됐었는데요,

C# 12 - 컬렉션 식(Collection Expressions)
; https://www.sysnet.pe.kr/2/0/13415

위의 글에서도 설명했듯이, 사용자 정의 타입에 대한 컬렉션 식을 지원하는 것이 가능합니다.


using System.Collections;

internal class Program
{
    static void Main(string[] args)
    {
        MyType list = [1, 2, 3, 4, 5, 6, 7, 8];
    }
}

public class MyType : IEnumerable // 1. IEnumerable을 구현한 경우
{
    List<int> _values { get; set; } = new List<int>();

    public IEnumerator GetEnumerator()
    {
        return _values.GetEnumerator();
    }

    public void Add(int value) // 2. "Add" 이름을 갖는 메서드 추가 (확장 메서드로 구현해도 OK)
    {
        _values.Add(value);
    }
}

그런데, 여기서 약간 아쉬운 점이 하나 있다면? "Add" 메서드를 이용한 초기화를 하다 보니 그다지 효율적인 코드가 나오지 않는다는 점입니다.

// 컬렉션 식을 사용한 경우, C# 컴파일러가 생성한 코드

MyType list = new MyType();
list.Add(1);
// ...[생략: 2, 3, 4, 5, 6, 7에 대해 동일하게 Add 호출]...
list.Add(8);

바로 저 문제를 해결하는 방법이 CollectionBuilder 특성을 통해 제공됩니다. 따라서 대충 다음과 같이 변경해 주면,

[CollectionBuilder(typeof(MyTypeCollectionBuilder), nameof(MyTypeCollectionBuilder.Create))]
public class MyType : IEnumerable<int>
{
    List<int> _values = new List<int>();

    public MyType(ReadOnlySpan<int> elems)
    {
        _values.AddRange(elems);
    }

    IEnumerator IEnumerable.GetEnumerator()
    {
        return _values.GetEnumerator();
    }

    public IEnumerator<int> GetEnumerator()
    {
        return ((IEnumerable<int>)_values).GetEnumerator();
    }
}

public static class MyTypeCollectionBuilder
{
    public static MyType Create(ReadOnlySpan<int> items) => new MyType(items);
}

C# 컴파일러는 이것을 인지하고, 컬렉션 식으로 초기화하는 코드를 다음과 같이 생성해 줍니다.

MyType list = MyTypeCollectionBuilder.Create(
    RuntimeHelpers.CreateSpan<int>(fieldof(...1~8까지의 숫자를 포함하고 있는 바이너리 영역의 위치...).FieldHandle)
);

보는 바와 같이 컬렉션 식으로 개발자가 명시한 "[1, 2, 3, 4, 5, 6, 7, 8]" 배열 값을 컴파일 시에 미리 바이너리에 저장한 영역으로 대신하고, 그 저장소 공간을 ReadOnlySpan<int>로 받아 사용자가 원하는 코드(위의 경우 AddRange)로 바로 초기화하게 됩니다.

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




참고로, CollectionBuilder로 지정하는 타입은 꼭 분리할 필요까진 없습니다. 즉, 위의 코드를 다음과 같이 하나의 타입에 정의해도 무방합니다.

[CollectionBuilder(typeof(MyType), nameof(Create))]
public class MyType : IEnumerable<int>
{
    List<int> _values = new List<int>();

    public MyType(ReadOnlySpan<int> elems)
    {
        _values.AddRange(elems);
    }

    IEnumerator IEnumerable.GetEnumerator()
    {
        return _values.GetEnumerator();
    }

    public IEnumerator<int> GetEnumerator()
    {
        return ((IEnumerable<int>)_values).GetEnumerator();
    }

    public static MyType Create(ReadOnlySpan<int> items) => new MyType(items);
}

관심 있으신 분들은 아래의 글도 읽어보시고. ^^

Adding support for collection expressions to your own types
; https://andrewlock.net/behind-the-scenes-of-collection-expressions-part-5-adding-support-for-collection-expressions-to-your-own-types/




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 8/6/2024]

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

비밀번호

댓글 작성자
 




[1]  2  3  4  5  6  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13935정성태5/20/2025309파이썬 - pip 사용 시 "ImportError: cannot import name 'html5lib' from 'pip._vendor'" 오류
13934정성태5/20/2025338스크립트: 77. 파이썬 - 'urllib.request' 모듈의 명시적/암시적 로딩 차이
13933정성태5/19/2025467오류 유형: 956. Visual Studio 2022가 17.12 버전부터 업데이트 되지 않는다면?
13932정성태5/18/2025516스크립트: 76. 파이썬 - Version 문자열 다루기(semver 패키지)
13931정성태5/17/2025907스크립트: 75. 파이썬 - Cython 기본 예제 및 컴파일
13930정성태5/17/2025831개발 환경 구성: 744. 파이썬 - Windows embeddable package 환경에서 외부 패키지 사용하는 방법(ex: UFO² 환경 구성)
13929정성태5/16/2025998오류 유형: 955. 파이썬 - "Windows embeddable package" REPL 환경에서 "NameError: name 'exit' is not defined"
13928정성태5/15/20251150오류 유형: 954. UFO² - "'Invalid URL (POST /v1/chat/completions/chat/completions)'"
13927정성태5/15/20251162오류 유형: 953. OpenAI - The API request of HOST_AGENT failed: OpenAI API request exceeded rate limit: Error code: 429
13926정성태5/14/20251574개발 환경 구성: 743. LLM과 윈도우의 만남 - Desktop AgentOS UFO² 기본 환경 구성
13925정성태5/12/20251660닷넷: 2333. C# - (Console 유형의 프로젝트에서) Clipboard 연동파일 다운로드1
13924정성태5/8/20251428닷넷: 2332. C# - (JetBrains Omea Reader 대상으로) 런타임 시에 메서드 가로채기 [2]파일 다운로드1
13923정성태5/5/20251221스크립트: 74. 파이썬 - C# - Python.NET의 RunSimpleScript, Exec, Eval 차이점파일 다운로드1
13922정성태5/3/20251350스크립트: 73. 파이썬 - Windows embeddable package 버전에서 tkinter 환경 구성
13921정성태5/3/20251609오류 유형: 952. 듀얼 채널 메모리 정렬을 지키지 않은 컴퓨터의 Windows 비정상 종료 현상(Blue Screen) [2]
13920정성태5/3/20251652오류 유형: 951. Typed DataSet 생성 중 "Failed to open a connection to the database" 오류
13919정성태5/2/20251519VS.NET IDE: 201. C# - Typed DataSet(XSD)를 위한 연결 문자열 암호화 [1]파일 다운로드1
13918정성태5/2/20251622VS.NET IDE: 200. C# - app.config 파일의 출력을 Configuration(Debug/Release)에 따라 제어하는 방법파일 다운로드1
13917정성태4/30/20251314VS.NET IDE: 199. Directory.Build.props에 정의한 속성에 대해 Condition 제약으로 값을 변경하는 방법
13916정성태4/23/20251117디버깅 기술: 221. WinDbg 분석 사례 - ASP.NET HttpCookieCollection을 다중 스레드에서 사용할 경우 무한 루프 현상 - 두 번째 이야기
13915정성태4/13/20252426닷넷: 2331. C# - 실행 시에 메서드 가로채기 (.NET 9)파일 다운로드1
13914정성태4/11/20252809디버깅 기술: 220. windbg 분석 사례 - x86 ASP.NET 웹 응용 프로그램의 CPU 100% 현상 (4)
13913정성태4/10/20251654오류 유형: 950. Process Explorer - 64비트 윈도우에서 32비트 프로세스의 덤프를 뜰 때 "Error writing dump file: Access is denied." 오류
13912정성태4/9/20251385닷넷: 2330. C# - 실행 시에 메서드 가로채기 (.NET 5 ~ .NET 8)파일 다운로드1
13911정성태4/8/20251670오류 유형: 949. WinDbg - .NET Core/5+ 응용 프로그램 디버깅 시 sos 확장을 자동으로 로드하지 못하는 문제
[1]  2  3  4  5  6  7  8  9  10  11  12  13  14  15  ...