Microsoft MVP성태의 닷넷 이야기
.NET Framework: 247. LINQ에서의 Max 기능 구현 [링크 복사], [링크+제목 복사],
조회: 33159
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일

LINQ에서의 Max 기능 구현


이 글은, 다음의 질문 때문에 씌여진 것입니다.

LINQ 가장 큰 값 하나만 가져오기.
; http://www.devpia.com/MAEUL/Contents/Detail.aspx?BoardID=17&MAEULNO=8&no=140468&ref=140468&page=1

자... 문제를 여기다 다시 써볼까요?

Program tt = new Program();

tt.data = new List<FileInfo>();
tt.data.Add(new FileInfo { id = 1, filename = "AUpIs80.log", sended = true });
tt.data.Add(new FileInfo { id = 2, filename = "adminxplore.exe", sended = false });
tt.data.Add(new FileInfo { id = 3, filename = "LMDebug.log", sended = false });
tt.data.Add(new FileInfo { id = 4, filename = "LMDebug1.log", sended = false });
tt.data.Add(new FileInfo { id = 5, filename = "LMDebug1.log", sended = false });

var linq = from a in tt.data
    orderby a.id
    where a.sended == false
    select a.id;

안녕하세요 저런 코드에서 id가 가장 크거나 작은 것 하나만 셀렉트 하는 방법이 없나요? Max를 이용 하면 될거 같은데 쉽지 않네요...
그리고 저렇게 var linq에 담게 되면 실제 메모리에 새로 만들어 지는건가요? 아니면 LIST data의 주소 값만 참조 하는건가요?

우선, 원하는 값이 무엇이냐에 따라 달라집니다. 간단하게는 가장 큰 "id" 값을 원할 수 있을 텐데요. 그런 경우에는 굳이 LINQ까지 갈 필요없이 Max 확장 함수를 이용해서 다음과 같이 구할 수 있습니다.

var max = tt.data.Where(a => a.sended == false).Max(a => a.id);

결과: max == 5

그런데, max == 5인 FileInfo 개체를 원한다면 어떻게 해야 할까요?

우선, 마이크로소프트의 예제 코드에 소개된 방식을 이용하여,

How to: Find the Maximum Value in a Numeric Sequence (LINQ to SQL)
; https://docs.microsoft.com/en-us/dotnet/framework/data/adonet/sql/linq/find-the-maximum-value-in-a-numeric-sequence

다음과 같이 구현되어질 수 있습니다.

var linq = from a in tt.data
            where a.sended == false && a.id == tt.data.Max(b => b.id)
            select a;

Console.WriteLine(linq.First().id);

결과야 나오긴 했지만, LINQ 구문을 잠시 살펴보면 매우 비효율적이라는 것을 알 수 있습니다. 왜냐하면, tt.data.Max(...)는 data에 있는 모든 요소들을 열람하게 되는데 그것의 실행 횟수가 역시 data 요소 수만큼 발생하기 때문입니다.

따라서, 이러한 쓸데없는 오버헤드를 줄이기 위해서는 다음과 같이 바꿔주는 것이 좋습니다.

var max = tt.data.Where(a => a.sended == false).Max(a => a.id);

var linq2 = from a in tt.data
            where a.sended == false && a.id == max
            select a;

그런데... 왠지 너무 길다는 생각이 안드세요? 차라리 첫 번째 "var max = ...Max(...)" 구문에서 객체값을 반환해 준다면 좋겠다는 생각이 듭니다. 아쉽게도 기본 제공되는 Max 확장 함수에서는 그에 전달된 람다식의 값을 반환하게 되어 있기 때문에 약간의 변형을 해주어야 합니다.

이를 위해서 Max에 대한 별도의 확장 함수를 제공해 주면 되는데, 이에 대해서는 다음의 글에서 자세하게 설명해 주고 있습니다.

Custom C# 3.0 LINQ Max Extension Method
; http://geekswithblogs.net/michelotti/archive/2009/02/06/custom-c-3.0-linq-max-extension-method.aspx

위의 글에 따라 Max 함수를 만들어 주면,

public static class ExtensionMethod
{
    public static FileInfo Max<TCompare>(this IEnumerable<FileInfo> collection, Func<FileInfo, TCompare> func) where TCompare : IComparable<TCompare>
    {
        FileInfo maxItem = null;
        TCompare maxValue = default(TCompare);

        foreach (var item in collection)
        {
            TCompare temp = func(item);
            if (maxItem == null || temp.CompareTo(maxValue) > 0)
            {
                maxValue = temp;
                maxItem = item;
            }
        }

        return maxItem;
    }
}

이제 다음과 같이 자연스럽게 Max 값을 구해낼 수 있습니다.

var max = tt.data.Where(a => a.sended == false).Max(a => a.id);
Console.WriteLine(max.id);




그런데, 한번 더 생각해 볼까요?

확장 함수 없이, 애당초 다음과 같은 구문으로 max를 구했으면 더 편했을 거 아닌가요?

var max = tt.data.Where(a => a.sended == false).Max(a => a.id);

==> var max = tt.data.Where(a => a.sended == false).Max(a => a);

실제로 위의 구문을 실행하면 다음과 같은 오류가 발생합니다.

Unhandled Exception: System.ArgumentException: At least one object must implement IComparable.
   at System.Collections.Comparer.Compare(Object a, Object b)
   at System.Collections.Generic.ObjectComparer`1.Compare(T x, T y)
   at System.Linq.Enumerable.Max[TSource](IEnumerable`1 source)
   at System.Linq.Enumerable.Max[TSource,TResult](IEnumerable`1 source, Func`2 selector)
   at ConsoleApplication1.Program.Main(String[] args) in D:\...[생략]...\Program.cs:line 36

아하... 그저 FileInfo 개체에 비교가 가능하도록 만들어주면 되겠군요.

public class FileInfo : IComparable
{
    public int id { get; set; }
    public string filename { get; set; }
    public bool sended { get; set; }

    public int CompareTo(object other)
    {
        FileInfo second = other as FileInfo;
        if (second == null)
        {
            return 1;
        }

        if (second.id > id) return -1;
        if (second.id == id) return 0;
        return 1;
    }
}

var max = tt.data.Where(a => a.sended == false).Max(a => a); // 정상적으로 실행됨.
Console.WriteLine(max.id);




이 정도면, 웬만큼 Max 관련해서는 의문점이 풀렸겠지요. ^^ 그럼, 부가적으로 마지막 질문으로 넘어가 볼까요?

그리고 저렇게 var linq에 담게 되면 실제 메모리에 새로 만들어 지는건가요? 아니면 LIST data의 주소 값만 참조 하는건가요?

이에 대해서는 간단하게 테스트 해서 알아보는 방법이 있습니다.

새롭게 메모리가 할당되는 것이라면, 기존 개체의 값을 바꿨다고 해서 LINQ 계산값이 바뀌지는 않았을 것이라는 간단한 원칙을 이용하면 되는 것이지요. ^^

var max4 = tt.data.Where(a => a.sended == false).Max(a => a);

tt.data[4].filename = "test";

Console.WriteLine(max4.filename);

출력: test

결과를 보니, tt.data 목록 안에 있는 개체의 참조값임을 알 수 있습니다. 메모리 할당이 아닌 참조값만 넘어왔다는 것입니다.

첨부된 파일은 위의 코드를 포함한 예제 프로젝트입니다.




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







[최초 등록일: ]
[최종 수정일: 6/26/2021]

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

비밀번호

댓글 작성자
 



2011-10-13 10시50분
[강덕훈] 와우~ 감사합니다. 한시간 넘게 곰곰히 생각 할 수 있는 답변이었습니다.

IComparable 에 대해서 확실히 배웠습니다.

컬럭센 관련 클래스에 Max 함수를 호출해도 컴파일은 되는데 런타임에서 에러가 나는게 신기하네요.

템플릿 클래스(FileInfo) 가 IComparable 를 임플리먼트 하지 않았다는걸 컴파일 단계에서 알려주면 더 좋을텐데 말이죠...

아마 제너릭 개념이 추가되어서 컴파일러가 어려움이 있는 듯 한데...

정말 많이 배우고 갑니다. 아 그리고

이 글의 맨 처음 부분에
var max = tt.data.Where(a => a.sended == false).Max(a => a.id);

결과: max == 5

이 부분도 런타임 오류가 나네요 ^^
[guest]
2011-10-13 01시18분
max에서 IComparable이 없다고 컴파일 단계에서 알려주지 않는 것은, Max 확장 함수에서 객체가 아닌 함수를 받기 때문입니다. 함수 자체는 IComparable 구현 여부를 알 수 없지요. ^^

그리고, 런타임 오류가 난다는 코드는 제가 이 글에 실린 예제에서 발생하는 것인가요? 이상한데요. 제가 첨부한 소스에서는 런타임 오류가 발생하지 않습니다. 혹시, 현재 런타임 오류가 발생하는 간략화된 소스를 올려주시겠어요? ^^
정성태
2011-10-13 02시36분
[강덕훈] 아~ 그렇군요.. 잘 배우고 갑니다. 아직 부족한게 너무 많네요 ㅋ


아뇨 예제는 IComparable 를 구현 했으니 오류가 안납니다.
이 글의 순서상 처음엔 IComparable를 구현 안했으니 "결과: max == 5" 안나온다는 거죠 ^^
[guest]
2011-10-13 02시45분
음... 이상한데요. ^^
분명히, IComparable 구현 여부와 상관없이 아래의 구문은 예외 없이 잘 실행됩니다.

var max = tt.data.Where(a => a.sended == false).Max(a => a.id);
결과: max == 5
정성태
2011-10-13 03시08분
max에서 IComparable에 대한 컴파일 단계의 오류에 대해서 좀 더 생각해 보았는데... 제가 잘못 대답을 했군요. ^^ 어차피 함수를 받긴 하지만, generic에서 받은 T에 대해 where 구문을 주면 되기 때문에 가능하군요.

가령 다음과 같은 코드입니다. (아래의 코드를 담아서, 첨부된 예제 프로젝트의 코드를 업데이트 했습니다.)

        public static TSource Max4<TSource>(this IEnumerable<TSource> collection, Func<TSource, TSource> func)
            where TSource : IComparable<TSource>
        {
            Console.WriteLine("Max4(TSource, TCompare) called!");

            TSource maxItem = default(TSource);
            TSource maxValue = default(TSource);

            foreach (var item in collection)
            {
                TSource temp = func(item);
                if (maxItem == null || temp.CompareTo(maxValue) > 0)
                {
                    maxValue = temp;
                    maxItem = item;
                }
            }

            return maxItem;
        }
정성태
2011-10-13 05시42분
[강덕훈] 아~ ㅋㅋ 감사합니다.

c#에 입문 한지 얼마 안되서 궁금한게 너무 많네요 ㅋ 하루에도 몇가지씩 생기네요

자주 방문 하겠습니다. ㅋ
[guest]
2012-03-06 09시47분
[김성일] 좋은글 감사합니다.~~~
[guest]
2023-04-26 09시54분
[밤밤] 내용 감사합니다
[guest]
2023-04-26 11시56분
@밤밤 귀찮게 고친 티가 나는군요. ^^; (퇴근 후에 수정하면 이후부터는 덧글이 안 될 것입니다.)
정성태
2023-04-26 08시18분
.NET 6 BCL부터 본문에 해당하는 목적의 LINQ 확장 메서드(MaxBy, MinBy)가 추가됐습니다.

Bite-Size .NET 6 - MaxBy() and MinBy() in LINQ
; https://exceptionnotfound.net/bite-size-dotnet-6-maxby-and-minby-in-linq/
정성태

... 76  77  78  79  [80]  81  82  83  84  85  86  87  88  89  90  ...
NoWriterDateCnt.TitleFile(s)
11936정성태6/10/201918367Math: 58. C# - 최소 자승법의 1차, 2차 수렴 그래프 변화 확인 [2]파일 다운로드1
11935정성태6/9/201919935.NET Framework: 843. C# - PLplot 출력을 파일이 아닌 Window 화면으로 변경
11934정성태6/7/201921271VC++: 133. typedef struct와 타입 전방 선언으로 인한 C2371 오류파일 다운로드1
11933정성태6/7/201919607VC++: 132. enum 정의를 C++11의 enum class로 바꿀 때 유의할 사항파일 다운로드1
11932정성태6/7/201918786오류 유형: 544. C++ - fatal error C1017: invalid integer constant expression파일 다운로드1
11931정성태6/6/201919300개발 환경 구성: 441. C# - CairoSharp/GtkSharp 사용을 위한 프로젝트 구성 방법
11930정성태6/5/201919831.NET Framework: 842. .NET Reflection을 대체할 System.Reflection.Metadata 소개 [1]
11929정성태6/5/201919396.NET Framework: 841. Windows Forms/C# - 클립보드에 RTF 텍스트를 복사 및 확인하는 방법 [1]
11928정성태6/5/201918171오류 유형: 543. PowerShell 확장 설치 시 "Catalog file '[...].cat' is not found in the contents of the module" 오류 발생
11927정성태6/5/201919397스크립트: 15. PowerShell ISE의 스크립트를 복사 후 PPT/Word에 붙여 넣으면 한글이 깨지는 문제 [1]
11926정성태6/4/201919932오류 유형: 542. Visual Studio - pointer to incomplete class type is not allowed
11925정성태6/4/201919769VC++: 131. Visual C++ - uuid 확장 속성과 __uuidof 확장 연산자파일 다운로드1
11924정성태5/30/201921410Math: 57. C# - 해석학적 방법을 이용한 최소 자승법 [1]파일 다운로드1
11923정성태5/30/201921031Math: 56. C# - 그래프 그리기로 알아보는 경사 하강법의 최소/최댓값 구하기파일 다운로드1
11922정성태5/29/201918532.NET Framework: 840. ML.NET 데이터 정규화파일 다운로드1
11921정성태5/28/201924389Math: 55. C# - 다항식을 위한 최소 자승법(Least Squares Method)파일 다운로드1
11920정성태5/28/201916052.NET Framework: 839. C# - PLplot 색상 제어
11919정성태5/27/201920306Math: 54. C# - 최소 자승법의 1차 함수에 대한 매개변수를 단순 for 문으로 구하는 방법 [1]파일 다운로드1
11918정성태5/25/201921148Math: 53. C# - 행렬식을 이용한 최소 자승법(LSM: Least Square Method)파일 다운로드1
11917정성태5/24/201922133Math: 52. MathNet을 이용한 간단한 통계 정보 처리 - 분산/표준편차파일 다운로드1
11916정성태5/24/201919948Math: 51. MathNET + OxyPlot을 이용한 간단한 통계 정보 처리 - Histogram파일 다운로드1
11915정성태5/24/201923058Linux: 11. 리눅스의 환경 변수 관련 함수 정리 - putenv, setenv, unsetenv
11914정성태5/24/201922044Linux: 10. 윈도우의 GetTickCount와 리눅스의 clock_gettime파일 다운로드1
11913정성태5/23/201918762.NET Framework: 838. C# - 숫자형 타입의 bit(2진) 문자열, 16진수 문자열 구하는 방법파일 다운로드1
11912정성태5/23/201918726VS.NET IDE: 137. Visual Studio 2019 버전 16.1부터 리눅스 C/C++ 프로젝트에 추가된 WSL 지원
11911정성태5/23/201917491VS.NET IDE: 136. Visual Studio 2019 - 리눅스 C/C++ 프로젝트에 인텔리센스가 동작하지 않는 경우
... 76  77  78  79  [80]  81  82  83  84  85  86  87  88  89  90  ...