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

함수형 프로그래밍 개념 - 리스트 해석(List Comprehension)과 순수 함수

절차형 프로그래밍만 하다가 함수형 프로그래밍을 배우는 경우 용어에서 약간 멈칫하는 경우가 있습니다. 그냥 오늘은 혹시 도움이 되실까 싶어 2가지 정도 소개해 드릴까 합니다.

우선, 리스트 해석(List Comprehension)인데, 다음의 글을 보시면 이해가 화~~~악 됩니다.

Chapter 3. Comprehensions
; http://juehan.github.io/DiveIntoPython3_Korean_Translation/comprehensions.html

리스트 컴프리헨션(list comprehension)이란 특정 리스트의 각각의 원소에 어떤 함수를 적용한 후 그 결과를 받아 새로운 리스트로 만들 수 있는 아주 간편한 방법


위의 글에 소개된 파이썬 예제 코드를 보면 더욱 이해가 빠르겠지요.

a_list = [1, 9, 8, 4]
[elem * 2 for elem in a_list]  // a_list의 모든 요소에 * 2를 해서 새로운 리스트를 만듦

사실 C#에도 LINQ가 나오면서 어느 정도 이런 구문이 가능합니다. 그렇습니다. SELECT 구문이 바로 그것입니다.

List<int> list = new List<int>() { 1, 2, 3, 4 };
            
var list2 = from elem in list select elem * 2;

foreach (var item in list2)
{
    Console.WriteLine(item);
}

Entity Framework이 하도 유명해서 LINQ를 데이터베이스 접근 용도로만 아실 수 있는데, 엄밀히 LINQ는 C#에 어느 정도의 함수형 프로그래밍 요소를 가져온 기여도 무시할 수 없습니다.

이 외에, "Chapter 3. Comprehensions" 글을 보면 "comprehension"이란 단어가 컬렉션의 요소에 어떤 함수를 적용한 후 새로운 컬렉션을 반환하는 일반적인 용도로 쓰인다는 것을 알 수 있습니다. 즉, "Dictionary Comprehension" 등의 표현도 있다는 것이지요.




그다음, 순수 함수(Pure function)가 있습니다. "프로그래밍 클로저: Lisp" 책에 보면 순수 함수는 말 그대로 부수 효과(side effect)가 전혀 없는 함수를 말하며, 즉 오로지 인자에만 의존해서 결과가 만들어지고 반환 값으로만 외부 세계에 영향을 준다고 합니다. 이렇게 말하면 ^^ 잘 이해가 안되죠? 역시 코드를 곁들여 설명해 보겠습니다.

간단하게 예를 들어, 다음의 경우는 순수 함수가 아닙니다.

int GeneralMethod(int n)
{
    return n * DateTime.Now.Ticks;
}

GeneralMethod에 n == 1을 넣었을 때 항상 같은 값이 산출되는 것이 아니기 때문입니다. 순수 함수는, 언제나 동일한 입력 값이면 동일한 출력 값을 반환해야 합니다.

또한, 객체 내부 상태를 변경해서도 안됩니다. 이것은 어찌 보면 당연한 조건입니다. 객체의 내부 값을 변경하는 경우 해당 함수를 호출할 때마다 상태가 변경되므로 입력 값에 대한 출력 값이 '항상' 일치하지 않을 수 있기 때문입니다. (C++의 경우에는 const 함수가 적당한 예가 되겠군요.)

그런 의미에서 우리가 알고 있는 수학 함수라는 것들은 모두 '순수 함수'에 속합니다.

int a = Math.Abs(-5);

위의 C# 메서드는 5를 반환합니다. -5 입력 값이 들어오면, 무조건, 항상, 언제나 한결 같이 5를 반환합니다. 이것이 순수 함수입니다.

자연스럽게 순수 함수가 만연하게 되는 함수형 프로그래밍은 이런 특성으로 인해 절차형에 비해 호출에 대한 최적화가 가능합니다. 왜냐하면, Math.Abs 함수의 입력으로 -5가 들어왔다면 최초 한번은 해당 함수의 전체 코드를 실행해야 하지만, 이후부터는 입력값 -5에 대해 무조건 5를 반환하면 되기 때문입니다.

실제로 Clojure 언어에는 메모이제이션(memoization)이라는 구문이 제공됩니다. 예를 들어, 1초의 지연 시간을 일부러 포함한 다음의 clojure 함수를 정의한 후,

user=> (defn delay-print [] (Thread/sleep 1000) (println "done") 100)
#'user/delay-print

이 함수를 memoize 함수를 통해 메모이제이션을 수행하도록 할 수 있습니다.

user=> (def delay-print (memoize delay-print))
#'user/delay-print

그럼, delay-print 함수를 최초 실행했을 때는 1초의 지연 시간을 갖고 println 함수도 수행한 후 100의 정수값을 반환하지만,

user=> (delay-print)
done
100

이후에 다시 실행했을 때는 1초의 지연시간도, println 함수 수행도 없이 그냥 곧바로 100의 값을 반환합니다.

user=> (delay-print)
100

이걸 테스트 하다보면 nCr 조합의 값을 재귀호출을 통해 얻어내던 것이 생각납니다. ^^

동전을 여러 더미로 나누는 경우의 수 세기(Partition Number) - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/1719

위의 경우 C#으로 만든 이항 계수 재귀 함수가 가장 낮은 성능을 보여줬는데요. clojure의 경우 다음과 같이 memoize를 한번 해주면,

user=> (defn BC [n, r] (cond (or (= r 0) (= r n)) 1
                     :else (+ (BC (- n 1) (- r 1)) (BC (- n 1) r))
                )
)
#'user/BC

user=> (def BC (memoize BC))
#'user/BC

결과가 순식간에 출력됩니다.

user=> (time (BC 30 12))
"Elapsed time: 0.67197 msecs"
86493225

// 위의 time 함수는 수행 시간을 재주는 기능을 합니다.
// 만약, memoize를 하지 않으면 (제 컴퓨터 기준으로) 수행 시간이 15초 정도 소요됩니다.

아니, 그런데 위의 경우에는 (BC 30 12) 호출을 두 번째 한 것도 아닌데 어떻게 저리 빠른 실행 결과를 보여줄 수 있을까요? 왜냐하면, BC 함수가 재귀 호출이 되므로 내부의 호출 결과들이 모두 메모이제이션되기 때문에 내부 BC 함수 수행 중에 캐시된 결과값이 도움이 된 경우가 있었기 때문입니다.

만약 C#으로 순수 함수의 성능을 높이려면 일부러 결과를 캐시(cache)하는 작업을 해줘야 하는데 clojure의 경우 단순하게 memoize 함수의 사용으로 그것이 해결됩니다. (실제로 "동전을 여러 더미로 나누는 경우의 수 세기(Partition Number) - 두 번째 이야기" 글의 "Partition Number" 예제에서는 EulerFunction 타입의 P 메서드에 캐시를 추가해서 수행 속도를 높였습니다.)

그러고 보니 '순식간'에라는 표현으로 과거에 썼던 글이 하나 생각나는 군요.

함수형 언어의 코드가 그렇게 빠를까?
; https://www.sysnet.pe.kr/2/0/1324

솔직히, 위의 글을 쓰던 순간에는 함수형 언어에 대한 이해가 많이 부족했었는데요. 이제는 왜 함수형 언어들이 빠를 수가 있을까... 하는 것이 이해가 됩니다. 또한 위의 글에서 테스트 했던 haskell과 F#의 메모리 사용량도 이해가 됩니다. 어쩌면, 나름대로의 함수형 언어라는 특징을 기반으로 내부적인 수행 최적화를 더 할 수도 있었을 테고 그 일정 부분에는 메모이제이션과 같은 캐시 역할도 수반되었을지도 모르겠습니다.

암튼, 요즘 몇 가지 언어를 쭈욱 훑어보다 보니 이해의 폭이 넓어져서 좋은 것 같습니다. ^^




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







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

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

비밀번호

댓글 작성자
 



2014-08-18 02시34분
파이썬 - 데코레이터를 통한 memoization
; http://soooprmx.com/wp/archives/5149
정성태
2023-04-06 09시48분
정성태

... 16  17  [18]  19  20  21  22  23  24  25  26  27  28  29  30  ...
NoWriterDateCnt.TitleFile(s)
13203정성태12/22/20224872.NET Framework: 2081. C# Interop 예제 - (LSA_UNICODE_STRING 예제로) 구조체를 C++에 전달하는 방법파일 다운로드1
13202정성태12/21/20225260기타: 84. 직렬화로 설명하는 Little/Big Endian파일 다운로드1
13201정성태12/20/20225822오류 유형: 835. PyCharm 사용 시 C 드라이브 용량 부족
13200정성태12/19/20224758오류 유형: 834. 이벤트 로그 - SSL Certificate Settings created by an admin process for endpoint
13199정성태12/19/20224942개발 환경 구성: 656. Internal Network 유형의 스위치로 공유한 Hyper-V의 VM과 호스트가 통신이 안 되는 경우
13198정성태12/18/20224884.NET Framework: 2080. C# - Microsoft.XmlSerializer.Generator 처리 없이 XmlSerializer 생성자를 예외 없이 사용하고 싶다면?파일 다운로드1
13197정성태12/17/20224670.NET Framework: 2079. .NET Core/5+ 환경에서 XmlSerializer 사용 시 System.IO.FileNotFoundException 예외 발생하는 경우파일 다운로드1
13196정성태12/16/20224875.NET Framework: 2078. .NET Core/5+를 위한 SGen(Microsoft.XmlSerializer.Generator) 사용법
13195정성태12/15/20225323개발 환경 구성: 655. docker - bridge 네트워크 모드에서 컨테이너 간 통신 시 --link 옵션 권장 이유
13194정성태12/14/20225463오류 유형: 833. warning C4747: Calling managed 'DllMain': Managed code may not be run under loader lock파일 다운로드1
13193정성태12/14/20225579오류 유형: 832. error C7681: two-phase name lookup is not supported for C++/CLI or C++/CX; use /Zc:twoPhase-
13192정성태12/13/20225634Linux: 55. 리눅스 - bash shell에서 실수 연산
13191정성태12/11/20226514.NET Framework: 2077. C# - 직접 만들어 보는 SynchronizationContext파일 다운로드1
13190정성태12/9/20227007.NET Framework: 2076. C# - SynchronizationContext 기본 사용법파일 다운로드1
13189정성태12/9/20227897오류 유형: 831. Visual Studio - Windows Forms 디자이너의 도구 상자에 컨트롤이 보이지 않는 문제
13188정성태12/9/20226468.NET Framework: 2075. C# - 직접 만들어 보는 TaskScheduler 실습 (SingleThreadTaskScheduler)파일 다운로드1
13187정성태12/8/20226389개발 환경 구성: 654. openssl - CA로부터 인증받은 새로운 인증서를 생성하는 방법 (2)
13186정성태12/6/20224929오류 유형: 831. The framework 'Microsoft.AspNetCore.App', version '...' was not found.
13185정성태12/6/20225897개발 환경 구성: 653. Windows 환경에서의 Hello World x64 어셈블리 예제 (NASM 버전)
13184정성태12/5/20225075개발 환경 구성: 652. ml64.exe와 link.exe x64 실행 환경 구성
13183정성태12/4/20225099오류 유형: 830. MASM + CRT 함수를 사용하는 경우 발생하는 컴파일 오류 정리
13182정성태12/4/20225828Windows: 217. Windows 환경에서의 Hello World x64 어셈블리 예제 (MASM 버전)
13181정성태12/3/20225219Linux: 54. 리눅스/WSL - hello world 어셈블리 코드 x86/x64 (nasm)
13180정성태12/2/20225323.NET Framework: 2074. C# - 스택 메모리에 대한 여유 공간 확인하는 방법파일 다운로드1
13179정성태12/2/20224646Windows: 216. Windows 11 - 22H2 업데이트 이후 Terminal 대신 cmd 창이 뜨는 경우
13178정성태12/1/20225227Windows: 215. Win32 API 금지된 함수 - IsBadXxxPtr 유의 함수들이 안전하지 않은 이유파일 다운로드1
... 16  17  [18]  19  20  21  22  23  24  25  26  27  28  29  30  ...