Microsoft MVP성태의 닷넷 이야기
.NET Framework: 1089. C# - Indexer에 Range 및 람다 식을 이용한 필터 구현 [링크 복사], [링크+제목 복사],
조회: 15569
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일

C# - Indexer에 Range 및 람다 식을 이용한 필터 구현

파이썬 문법을 보니, 재미있는 활용 사례가 있군요.

import pandas as pd

""" book1.csv
name,kor,eng,mat
테스터,90,90,90
사용자,80,80,80
개발자,70,70,70
관리자,90,80,70
디자이너,60,50,50
"""

data = pd.read_csv('book1.csv')

data = data[data.mat > 70]
print(data)

""" 출력 결과
  name  kor  eng  mat
0  홍길동   90   90   90
1  임꺽정   80   80   80
5  테스터  100   50   90
"""

C#의 경우 indexer에 전통적으로 int나 string 정도만을 사용했었는데요, Python에 구현된 저 코드를 보니 틀에 박힌 생각이었던 것 같습니다. ^^

따라서, 그동안 LINQ 확장 메서드인 Where로 써 오던 것을 클래스 내부에 indexer를 구현해 제공하는 것도 나쁘진 않을 듯합니다.

using System;
using System.Collections.Generic;
using System.Linq;

class Program
{
    static void Main(string[] args)
    {
        MyClass mc = new MyClass(Enumerable.Range(0, 10));

        var items = mc[(elem) => elem < 5]; // 0, 1, 2, 3, 4,
    }
}

public class MyClass
{
    List<int> _list = new List<int>();

    public MyClass(IEnumerable<int> seq)
    {
        _list.AddRange(seq);
    }

    public IEnumerable<int> this[Func<int, bool> condition]
    {
        get
        {
            return _list.Where(condition);
        }
    }
}

파이썬과 비교해 약간 람다를 위한 코드가 더 들어가지만 그래도 논리 연산자를 사용할 상황이 되면 더 직관적이긴 합니다.

items = mc[(elem) => elem > 5 && elem < 8];

파이썬의 경우 이를 위해서는 numpy의 힘을 빌려야만 했죠.

import pandas as pd
import numpy as np

data = pd.read_csv('book1.csv')

data = data[np.logical_and(data.mat > 70, data.mat < 90)]
print(data)




기왕 하는 김에 slicing도 C# 8부터 추가된 Range를 통해 indexer에 추가하면 더 완벽해질 듯합니다. ^^

MyClass mc = new MyClass(Enumerable.Range(0, 10));

// python - mc[:3]
var result1 = mc[..3];      // 0, 1, 2

// python - mc[1:3]
var result2 = mc[1..3];     // 1, 2 

// python - mc[5:]
var result3 = mc[5..];      // 5, 6, 7, 8, 9

// python - mc[5:-1]
var result4 = mc[5..^1];    // 5, 6, 7, 8

// python - mc[:]
var result5 = mc[..];       // 0, 1, 2, 3, 4, 5, 6, 7, 8, 9

public class MyClass
{
    //...[생략]...

    public IEnumerable<int> this[Range range] // [range.Start, range.End) 범위 선택
    {
        get 
        {
            /* 이렇게 해도 되겠지만,
            int offset = range.Start.Value;

            int take = range.End.Value - range.Start.Value;
            if (range.End.IsFromEnd == true)
            {
                take = (_list.Count - range.End.Value - range.Start.Value);
            }
            */

            (int offset, int take) = range.GetOffsetAndLength(_list.Count);
            return _list.Skip(offset).Take(take);
        }
    }

    //...[생략]...
}

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




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







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

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

비밀번호

댓글 작성자
 



2024-02-19 09시50분
// GetEnumerator extensions for Range in C#
// https://twitter.com/okyrylchuk/status/1759327774328701044/photo/1

namespace ConsoleApp1;

internal class Program
{
    static void Main(string[] args)
    {
        foreach (int i in 3..5)
        {
            Console.WriteLine(i);
        }
        Console.WriteLine();
        foreach (int i in ..3)
        {
            Console.WriteLine(i);
        }
        Console.WriteLine();
        foreach (int i in ^3..)
        {
            Console.WriteLine(i);
        }
    }
}

public static class GetEnumeratorExtensions
{
    public static IEnumerator<int> GetEnumerator(this Range range)
    {
        if (range.Start.IsFromEnd)
        {
            return Enumerable.Range(range.End.Value, range.Start.Value - range.End.Value + 1).Reverse().GetEnumerator();

        }
        else
        {
            return Enumerable.Range(range.Start.Value, range.End.Value - range.Start.Value + 1).GetEnumerator();
        }
    }
}
정성태

... 31  32  33  34  35  36  37  38  39  40  41  [42]  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
12922정성태1/14/202216305개발 환경 구성: 625. AKS - Azure Kubernetes Service 생성 및 SLO/SLA 변경 방법
12921정성태1/14/202213219개발 환경 구성: 624. Docker Desktop에서 별도 서버에 설치한 docker registry에 이미지 올리는 방법
12920정성태1/14/202214610오류 유형: 786. Camtasia - An error occurred with the camera: Failed to Add Video Sampler.
12919정성태1/13/202214101Windows: 199. Host Network Service (HNS)에 의해서 점유되는 포트
12918정성태1/13/202214988Linux: 47. WSL - shell script에서 설정한 환경 변수가 스크립트 실행 후 반영되지 않는 문제
12917정성태1/12/202214345오류 유형: 785. C# - The type or namespace name '...' could not be found (are you missing a using directive or an assembly reference?)
12916정성태1/12/202213415오류 유형: 784. TFS - One or more source control bindings for this solution are not valid and are listed below.
12915정성태1/11/202214300오류 유형: 783. Visual Studio - We didn't find any interpreters
12914정성태1/11/202217785VS.NET IDE: 172. 비주얼 스튜디오 2022의 파이선 개발 환경 지원
12913정성태1/11/202218063.NET Framework: 1133. C# - byte * (바이트 포인터)를 FileStream으로 쓰는 방법 [1]
12912정성태1/11/202217416개발 환경 구성: 623. ffmpeg.exe를 사용해 비디오 파일의 이미지를 PGM(Portable Gray Map) 파일 포맷으로 출력하는 방법 [1]
12911정성태1/11/202213773VS.NET IDE: 171. 비주얼 스튜디오 - 더 이상 만들 수 없는 "ASP.NET Core 3.1 Web Application (.NET Framework)" 프로젝트
12910정성태1/10/202214954제니퍼 .NET: 30. 제니퍼 닷넷 적용 사례 (8) - CPU high와 DB 쿼리 성능에 문제가 함께 있는 사이트
12909정성태1/10/202215872오류 유형: 782. Visual Studio 2022 설치 시 "Couldn't install Microsoft.VisualCpp.Redist.14.Latest"
12908정성태1/10/202213130.NET Framework: 1132. C# - ref/out 매개변수의 IL 코드 처리
12907정성태1/9/202214853오류 유형: 781. (youtube-dl.exe) 실행 시 "This app can't run on your PC" / "Access is denied." 오류 발생
12906정성태1/9/202216911.NET Framework: 1131. C# - 네임스페이스까지 동일한 타입을 2개의 DLL에서 제공하는 경우 충돌을 우회하는 방법 [1]파일 다운로드1
12905정성태1/8/202216060오류 유형: 780. Could not load file or assembly 'Microsoft.VisualStudio.TextTemplating.VSHost.15.0, Version=16.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a' or one of its dependencies.
12904정성태1/8/202218016개발 환경 구성: 623. Visual Studio 2022 빌드 환경을 위한 github Actions 설정 [1]
12903정성태1/7/202217079.NET Framework: 1130. C# - ELEMENT_TYPE_INTERNAL 유형의 사용 예
12902정성태1/7/202215985오류 유형: 779. SQL 서버 로그인 에러 - provider: Shared Memory Provider, error: 0 - No process is on the other end of the pipe.
12901정성태1/5/202216871오류 유형: 778. C# - .NET 5+에서 warning CA1416: This call site is reachable on all platforms. '...' is only supported on: 'windows' 경고 발생
12900정성태1/5/202218962개발 환경 구성: 622. vcpkg로 ffmpeg를 빌드하는 경우 생성될 구성 요소 제어하는 방법
12899정성태1/3/202218523개발 환경 구성: 621. windbg에서 python 스크립트 실행하는 방법 - pykd (2)
12898정성태1/2/202219211.NET Framework: 1129. C# - ffmpeg(FFmpeg.AutoGen)를 이용한 비디오 인코딩 예제(encode_video.c) [1]파일 다운로드1
12897정성태1/2/202216365.NET Framework: 1128. C# - 화면 캡처한 이미지를 ffmpeg(FFmpeg.AutoGen)로 동영상 처리 [4]파일 다운로드1
... 31  32  33  34  35  36  37  38  39  40  41  [42]  43  44  45  ...