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)
11887정성태5/8/201921506.NET Framework: 829. C# - yield 문을 사용할 수 있는 메서드의 조건
11886정성태5/7/201919316오류 유형: 534. mstest.exe 실행 시 "Visual Studio Enterprise is required to execute the test." 오류 [2]
11885정성태5/7/201916262오류 유형: 533. mstest.exe 실행 시 "File extension specified '.loadtest' is not a valid test extension." 오류 발생
11884정성태5/5/201921053.NET Framework: 828. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 두 번째 이야기
11883정성태5/3/201926237.NET Framework: 827. C# - 인터넷 시간 서버로부터 받은 시간을 윈도우에 적용하는 방법파일 다운로드1
11882정성태5/2/201922527.NET Framework: 826. (번역글) .NET Internals Cookbook Part 11 - Various C# riddles파일 다운로드1
11881정성태4/28/201922638오류 유형: 532. .NET Core 프로젝트로 마이그레이션 시 "CS0579 Duplicate 'System.Reflection.AssemblyCompanyAttribute' attribute" 오류 발생
11880정성태4/25/201918491오류 유형: 531. 이벤트 로그 오류 - Task Scheduling Error: m->NextScheduledSPRetry 1547, m->NextScheduledEvent 1547
11879정성태4/24/201926908.NET Framework: 825. (번역글) .NET Internals Cookbook Part 10 - Threads, Tasks, asynchronous code and others파일 다운로드2
11878정성태4/22/201922692.NET Framework: 824. (번역글) .NET Internals Cookbook Part 9 - Finalizers, queues, card tables and other GC stuff파일 다운로드1
11877정성태4/22/201922751.NET Framework: 823. (번역글) .NET Internals Cookbook Part 8 - C# gotchas파일 다운로드1
11876정성태4/21/201921737.NET Framework: 822. (번역글) .NET Internals Cookbook Part 7 - Word tearing, locking and others파일 다운로드1
11875정성태4/21/201922791오류 유형: 530. Visual Studo에서 .NET Core 프로젝트를 열 때 "One or more errors occurred." 오류 발생
11874정성태4/20/201922973.NET Framework: 821. (번역글) .NET Internals Cookbook Part 6 - Object internals파일 다운로드1
11873정성태4/19/201921576.NET Framework: 820. (번역글) .NET Internals Cookbook Part 5 - Methods, parameters, modifiers파일 다운로드1
11872정성태4/17/201922356.NET Framework: 819. (번역글) .NET Internals Cookbook Part 4 - Type members파일 다운로드1
11871정성태4/16/201920932.NET Framework: 818. (번역글) .NET Internals Cookbook Part 3 - Initialization tricks [3]파일 다운로드1
11870정성태4/16/201919241.NET Framework: 817. Process.Start로 실행한 콘솔 프로그램의 출력 결과를 얻는 방법파일 다운로드1
11869정성태4/15/201925081.NET Framework: 816. (번역글) .NET Internals Cookbook Part 2 - GC-related things [2]파일 다운로드2
11868정성태4/15/201921068.NET Framework: 815. CER(Constrained Execution Region)이란?파일 다운로드1
11867정성태4/15/201920252.NET Framework: 814. Critical Finalizer와 SafeHandle의 사용 의미파일 다운로드1
11866정성태4/9/201923337Windows: 159. 네트워크 공유 폴더(net use)에 대한 인증 정보는 언제까지 유효할까요?
11865정성태4/9/201919162오류 유형: 529. 제어판 - C:\ProgramData\Microsoft\Windows\Start Menu\Programs\Administrative Tools is not accessible.
11864정성태4/9/201917852오류 유형: 528. '...' could be '0': this does not adhere to the specification for the function '...'
11863정성태4/9/201917746디버깅 기술: 127. windbg - .NET x64 EXE의 EntryPoint
11862정성태4/7/201920266개발 환경 구성: 437. .NET EXE의 ASLR 기능을 끄는 방법
... 76  77  78  79  80  81  [82]  83  84  85  86  87  88  89  90  ...