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

C# - Python range 함수 구현 (2) INumber<T>를 이용한 개선

다음의 기능이 나왔으니,

.NET 6 Preview 7에 추가된 숫자 형식에 대한 제네릭 연산 지원
; https://www.sysnet.pe.kr/2/0/12785

이제 Python의 Range 함수를,

C# - Python range 함수 구현
; https://www.sysnet.pe.kr/2/0/11908

더 매끄럽게 구현할 수 있습니다. ^^ 이렇게!

public static IEnumerable<T> Range<T>(T start, T stop, T step = default) where T: INumber<T>
{            
    if (step == default)
    {
        step = T.One;
    }

    if (start < stop && step > T.Zero)
    {
        for (var i = start; i < stop; i += step)
        {
            yield return i;
        }
    }
    else if (start > stop && step < T.Zero)
    {
        for (var i = start; i > stop; i += step)
        {
            yield return i;
        }
    }
}

dynamic을 쓸 필요도 없고 unmanaged 제약으로 unsafe를 쓸 필요도 없어졌습니다. 당연히 출력 결과도 기존의 코드와 완전히 동일합니다. ^^

static void Main(string[] args)
{
    {
        var expected = Extension.RangePython<int>(1, 10, 1); // 기존 방식
        var results = Extension.Range(1, 10); // 새로운 방식
        foreach (var item in results.Zip(expected, (first, second) => Tuple.Create(first, second)))
        {
            Console.WriteLine($"{item.Item1} == {item.Item2}: {item.Item1 == item.Item2}");
        }
    }

    Console.WriteLine();

    {
        var expected = Extension.RangePython<double>(0.1, .13, 0.001); // 기존 방식
        var results = Extension.Range(0.1, .13, 0.001); // 새로운 방식
        foreach (var item in results.Zip(expected, (first, second) => Tuple.Create(first, second)))
        {
            Console.WriteLine($"{item.Item1} == {item.Item2}: {item.Item1 == item.Item2}");
        }
    }
}

/* 출력 결과
1 == 1: True
2 == 2: True
3 == 3: True
4 == 4: True
5 == 5: True
6 == 6: True
7 == 7: True
8 == 8: True
9 == 9: True

0.1 == 0.1: True
0.101 == 0.101: True
0.10200000000000001 == 0.10200000000000001: True
0.10300000000000001 == 0.10300000000000001: True
0.10400000000000001 == 0.10400000000000001: True
0.10500000000000001 == 0.10500000000000001: True
0.10600000000000001 == 0.10600000000000001: True
0.10700000000000001 == 0.10700000000000001: True
0.10800000000000001 == 0.10800000000000001: True
0.10900000000000001 == 0.10900000000000001: True
0.11000000000000001 == 0.11000000000000001: True
0.11100000000000002 == 0.11100000000000002: True
0.11200000000000002 == 0.11200000000000002: True
0.11300000000000002 == 0.11300000000000002: True
0.11400000000000002 == 0.11400000000000002: True
0.11500000000000002 == 0.11500000000000002: True
0.11600000000000002 == 0.11600000000000002: True
0.11700000000000002 == 0.11700000000000002: True
0.11800000000000002 == 0.11800000000000002: True
0.11900000000000002 == 0.11900000000000002: True
0.12000000000000002 == 0.12000000000000002: True
0.12100000000000002 == 0.12100000000000002: True
0.12200000000000003 == 0.12200000000000003: True
0.12300000000000003 == 0.12300000000000003: True
0.12400000000000003 == 0.12400000000000003: True
0.12500000000000003 == 0.12500000000000003: True
0.12600000000000003 == 0.12600000000000003: True
0.12700000000000003 == 0.12700000000000003: True
0.12800000000000003 == 0.12800000000000003: True
0.12900000000000003 == 0.12900000000000003: True

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




그런데, 한 가지 아쉬운 것이 있군요. ^^

public static IEnumerable<T> Range<T>(T start, T stop, T step = default) where T: INumber<T>
{        
    if (step == default)
    {
        step = T.One;
    }
    
    // ...[생략]...
}

위의 코드에서 T step의 기본값을 default로 주었는데, 원래는 다음과 같이 설정할 수 있었으면 더 좋았을 것입니다.

public static IEnumerable<T> Range<T>(T start, T stop, T step = T.One) where T: INumber<T>

하지만, 메서드 선언에서의 문맥 단계에서는 T에 대한 풀이를 아직 제공하지 않는 것인지(혹은 못하는 것인지) 현재는 다음과 같은 컴파일 오류가 발생합니다.

error CS0103: The name 'T' does not exist in the current context

이것만 더해진다면 완벽하지 않을까... 싶군요. ^^




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







[최초 등록일: ]
[최종 수정일: 8/18/2021]

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

비밀번호

댓글 작성자
 



2021-09-07 11시00분
System.Range를 이용한 구현 방법을 소개하는데,

[C#] foreach에 Range 사용하기
; https://forum.dotnetdev.kr/t/c-foreach-range/1630

Making «foreach» loop as fast as «for» loop
; https://habr.com/en/post/575916/

표현이 더 간결해서 마음에 듭니다. ^^

foreach (var i in 1..1000)
{
    Console.WriteLine(i);
}

static class RangeExtension
{
    public static IEnumerator<int> GetEnumerator(this Range @this)
    {
        for (var i = @this.Start.Value; i < @this.End.Value; i++)
        {
            yield return i;
        }
    }
}
정성태

... 121  122  123  124  125  126  127  128  [129]  130  131  132  133  134  135  ...
NoWriterDateCnt.TitleFile(s)
1866정성태2/20/201519887오류 유형: 270. "aspnet_regiis -i" 실행 시 0x00000006 오류 해결 방법
1865정성태2/20/201521146.NET Framework: 499. 특정 닷넷 프레임워크 버전 이후부터 제공되는 타입을 사용해야 한다면?
1864정성태2/18/201526098.NET Framework: 498. C#으로 간단하게 만들어 본 ASCII Art 프로그램 [2]파일 다운로드1
1862정성태2/18/201530192.NET Framework: 497. .NET Garbage Collection에 대한 정리 [6]
1861정성태2/18/201525341.NET Framework: 496. 마우스 커서가 놓인 지점의 문자열 얻는 방법 [1]파일 다운로드1
1860정성태2/18/201525294.NET Framework: 495. CorElementType의 요소 값 설명파일 다운로드1
1859정성태2/17/201525748Windows: 106. 컴퓨터를 재부팅하면 절전(Power Saver) 전원 모드로 돌아가는 경우
1858정성태2/16/201535622Windows: 105. 자동으로 로그아웃/잠김 화면 상태로 전환된다면? [2]
1857정성태2/16/201523757.NET Framework: 494. 값(struct) 형식의 제네릭(Generic) 타입이 박싱되는 경우의 메타데이터 토큰 값파일 다운로드1
1856정성태2/15/201522568.NET Framework: 493. TypeRef 메타테이블에 등록되는 타입의 조건파일 다운로드1
1855정성태2/10/201521994개발 환경 구성: 256. WebDAV Redirector - Sysinternals 폴더 연결 시 "The network path was not found" 오류 해결 방법
1854정성태2/10/201522923Windows: 104. 폴더는 삭제할 수 없지만, 그 하위 폴더/파일은 생성/삭제/변경하는 보안 설정
1853정성태2/6/201553423웹: 29. 여신금융협회 웹 사이트의 "Netscape 6.0은 지원하지 않습니다." 오류 메시지 [5]
1852정성태2/5/201523927.NET Framework: 492. .NET CLR Memory 성능 카운터의 의미파일 다운로드1
1851정성태2/5/201524610VC++: 88. 하룻밤의 꿈 - 인텔 하스웰의 TSX Instruction 지원 [2]
1850정성태2/4/201545521Windows: 103. 작업 관리자에서의 "Commit size"가 가리키는 메모리의 의미 [4]
1849정성태2/4/201525054기타: 51. DropBox의 CPU 100% 현상 [1]파일 다운로드1
1848정성태2/4/201520677.NET Framework: 491. 닷넷 Generic 타입의 메타 데이터 토큰 값 알아내는 방법 [2]
1847정성태2/3/201524043기타: 50. C# - 윈도우에서 dropbox 동기화 폴더 경로 및 종료하는 방법
1846정성태2/2/201533203Windows: 102. 제어판의 프로그램 추가/삭제 항목을 수동으로 실행하고 싶다면? [1]
1845정성태1/26/201534326Windows: 101. 제어판의 "Windows 자격 증명 관리(Manage your credentials)"를 금지시키는 방법
1844정성태1/26/201531923오류 유형: 269. USB 메모리의 용량이 비정상적으로 보여진다면? [7]
1843정성태1/24/201523086VC++: 87. 무시할 수 없는 Visual C++ 런타임 함수 성능
1842정성태1/23/201546158개발 환경 구성: 255. 노트북 키보드에 없는 BREAK 키를 다른 키로 대체하는 방법
1841정성태1/21/201520549오류 유형: 268. Win32 핸들 관련 CLR4 보안 오류 사례
1840정성태1/8/201528920오류 유형: 267. Visual Studio - CodeLens 사용 시 CPU 100% 현상
... 121  122  123  124  125  126  127  128  [129]  130  131  132  133  134  135  ...