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분
정성태

1  2  3  4  5  6  7  8  9  10  11  12  [13]  14  15  ...
NoWriterDateCnt.TitleFile(s)
13316정성태4/10/20233645오류 유형: 854. docker-compose 시 "json.decoder.JSONDecodeError: Expecting value: line 1 column 1 (char 0)" 오류 발생
13315정성태4/10/20233911Windows: 245. Win32 - 시간 만료를 갖는 컨텍스트 메뉴와 윈도우 메시지의 영역별 정의파일 다운로드1
13314정성태4/9/20234065개발 환경 구성: 672. DosBox를 이용한 Turbo C, Windows 3.1 설치
13313정성태4/9/20234078개발 환경 구성: 671. Hyper-V VM에 Turbo C 2.0 설치 [2]
13312정성태4/8/20234160Windows: 244. Win32 - 시간 만료를 갖는 MessageBox 대화창 구현 (개선된 버전)파일 다운로드1
13311정성태4/7/20234610C/C++: 163. Visual Studio 2022 - DirectShow 예제 컴파일(WAV Dest)
13310정성태4/6/20234260C/C++: 162. Visual Studio - /NODEFAULTLIB 옵션 설정 후 수동으로 추가해야 할 library
13309정성태4/5/20234385.NET Framework: 2107. .NET 6+ FileStream의 구조 변화
13308정성태4/4/20234271스크립트: 47. 파이썬의 time.time() 실숫값을 GoLang / C#에서 사용하는 방법
13307정성태4/4/20234049.NET Framework: 2106. C# - .NET Core/5+ 환경의 Windows Forms 응용 프로그램에서 HINSTANCE 구하는 방법
13306정성태4/3/20233911Windows: 243. Win32 - 윈도우(cbWndExtra) 및 윈도우 클래스(cbClsExtra) 저장소 사용 방법
13305정성태4/1/20234295Windows: 242. Win32 - 시간 만료를 갖는 MessageBox 대화창 구현 (쉬운 버전)파일 다운로드1
13304정성태3/31/20234603VS.NET IDE: 181. Visual Studio - C/C++ 프로젝트에 application manifest 적용하는 방법
13303정성태3/30/20233934Windows: 241. 환경 변수 %PATH%에 DLL을 찾는 규칙
13302정성태3/30/20234552Windows: 240. RDP 환경에서 바뀌는 %TEMP% 디렉터리 경로
13301정성태3/29/20234651Windows: 239. C/C++ - Windows 10 Version 1607부터 지원하는 /DEPENDENTLOADFLAG 옵션파일 다운로드1
13300정성태3/28/20234351Windows: 238. Win32 - Modal UI 창에 올바른 Owner(HWND)를 설정해야 하는 이유
13299정성태3/27/20234096Windows: 237. Win32 - 모든 메시지 루프를 탈출하는 WM_QUIT 메시지
13298정성태3/27/20234068Windows: 236. Win32 - MessageBeep 소리가 안 들린다면?
13297정성태3/26/20234690Windows: 235. Win32 - Code Modal과 UI Modal
13296정성태3/25/20234058Windows: 234. IsDialogMessage와 협업하는 WM_GETDLGCODE Win32 메시지 [1]파일 다운로드1
13295정성태3/24/20234297Windows: 233. Win32 - modeless 대화창을 modal처럼 동작하게 만드는 방법파일 다운로드1
13294정성태3/22/20234465.NET Framework: 2105. LargeAddressAware 옵션이 적용된 닷넷 32비트 프로세스의 가용 메모리 - 두 번째
13293정성태3/22/20234517오류 유형: 853. dumpbin - warning LNK4048: Invalid format file; ignored
13292정성태3/21/20234670Windows: 232. C/C++ - 일반 창에도 사용 가능한 IsDialogMessage파일 다운로드1
13291정성태3/20/20235011.NET Framework: 2104. C# Windows Forms - WndProc 재정의와 IMessageFilter 사용 시의 차이점
1  2  3  4  5  6  7  8  9  10  11  12  [13]  14  15  ...