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;
        }
    }
}
정성태

... 76  [77]  78  79  80  81  82  83  84  85  86  87  88  89  90  ...
NoWriterDateCnt.TitleFile(s)
12103정성태1/7/202027795DDK: 8. Visual Studio 2019 + WDK Legacy Driver 제작- Hello World 예제 [1]파일 다운로드2
12102정성태1/6/202022380디버깅 기술: 152. User 권한(Ring 3)의 프로그램에서 _ETHREAD 주소(및 커널 메모리를 읽을 수 있다면 _EPROCESS 주소) 구하는 방법
12101정성태1/5/202024708.NET Framework: 876. C# - PEB(Process Environment Block)를 통해 로드된 모듈 목록 열람
12100정성태1/3/202021556.NET Framework: 875. .NET 3.5 이하에서 IntPtr.Add 사용
12099정성태1/3/202024751디버깅 기술: 151. Windows 10 - Process Explorer로 확인한 Handle 정보를 windbg에서 조회 [1]
12098정성태1/2/202024580.NET Framework: 874. C# - 커널 구조체의 Offset 값을 하드 코딩하지 않고 사용하는 방법 [3]
12097정성태1/2/202021901디버깅 기술: 150. windbg - Wow64, x86, x64에서의 커널 구조체(예: TEB) 구조체 확인
12096정성태12/30/201923316디버깅 기술: 149. C# - DbgEng.dll을 이용한 간단한 디버거 제작 [1]
12095정성태12/27/201925872VC++: 135. C++ - string_view의 동작 방식
12094정성태12/26/201924990.NET Framework: 873. C# - 코드를 통해 PDB 심벌 파일 다운로드 방법
12093정성태12/26/201923598.NET Framework: 872. C# - 로딩된 Native DLL의 export 함수 목록 출력파일 다운로드1
12092정성태12/25/201921362디버깅 기술: 148. cdb.exe를 이용해 (ntdll.dll 등에 정의된) 커널 구조체 출력하는 방법
12091정성태12/25/201925349디버깅 기술: 147. pdb 파일을 다운로드하기 위한 symchk.exe 실행에 필요한 최소 파일 [1]
12090정성태12/24/201924726.NET Framework: 871. .NET AnyCPU로 빌드된 PE 헤더의 로딩 전/후 차이점 [1]파일 다운로드1
12089정성태12/23/201922022디버깅 기술: 146. gflags와 _CrtIsMemoryBlock을 이용한 Heap 메모리 손상 여부 체크
12088정성태12/23/201921926Linux: 28. Linux - 윈도우의 "Run as different user" 기능을 shell에서 실행하는 방법
12087정성태12/21/201922029디버깅 기술: 145. windbg/sos - Dictionary의 entries 배열 내용을 모두 덤프하는 방법 (do_hashtable.py) [1]
12086정성태12/20/201925640디버깅 기술: 144. windbg - Marshal.FreeHGlobal에서 발생한 덤프 분석 사례
12085정성태12/20/201923984오류 유형: 586. iisreset - The data is invalid. (2147942413, 8007000d) 오류 발생 - 두 번째 이야기 [1]
12084정성태12/19/201923594디버깅 기술: 143. windbg/sos - Hashtable의 buckets 배열 내용을 모두 덤프하는 방법 (do_hashtable.py) [1]
12083정성태12/17/201926036Linux: 27. linux - lldb를 이용한 .NET Core 응용 프로그램의 메모리 덤프 분석 방법 [2]
12082정성태12/17/201924580오류 유형: 585. lsof: WARNING: can't stat() fuse.gvfsd-fuse file system
12081정성태12/16/201927324개발 환경 구성: 465. 로컬 PC에서 개발 중인 ASP.NET Core 웹 응용 프로그램을 다른 PC에서도 접근하는 방법 [5]
12080정성태12/16/201923387.NET Framework: 870. C# - 프로세스의 모든 핸들을 열람
12079정성태12/13/201926043오류 유형: 584. 원격 데스크톱(rdp) 환경에서 다중 또는 고용량 파일 복사 시 "Unspecified error" 오류 발생
12078정성태12/13/201926245Linux: 26. .NET Core 응용 프로그램을 위한 메모리 덤프 방법 [3]
... 76  [77]  78  79  80  81  82  83  84  85  86  87  88  89  90  ...