Microsoft MVP성태의 닷넷 이야기
개발 환경 구성: 241. 책 "프로그래밍 클로저: Lisp"을 읽고 나서. [링크 복사], [링크+제목 복사],
조회: 23814
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

책 "프로그래밍 클로저: Lisp"을 읽고 나서.

Clojure 개념만이라도 한번 익혀두려고 개발환경 설정하고 ^^

Clojure 언어의 윈도우 개발환경 설정
; https://www.sysnet.pe.kr/2/0/1693

책 한권 읽어봤는데요.

프로그래밍 클로저: Lisp 
; http://www.yes24.com/24/Goods/3907543?Acode=101

위의 책을 읽으면서 거친 약간의 시행착오랄까.... 그런 것을 정리해 봅니다.

우선, 위의 책에서는 "clojure-contrib" 라이브러리에 있는 함수를 간혹 사용하는데요. 이상하게도 제가 설치한 clojure-contrib 바이너리는 몇 가지 함수들이 동작하지 않았습니다. 검색을 해보니,

Why does my Clojure import fail?
; http://stackoverflow.com/questions/7591649/why-does-my-clojure-import-fail

그 이유가 나오는데요.

In addition to the answers saying that contrib 1.1 is incompatible with clojure 1.3

Taken from here:
Versions of clojure-contrib are matched to versions of Clojure.

If you are using Clojure 1.0, use clojure-contrib 1.0.
If you are using Clojure 1.1, use clojure-contrib 1.1.
If you are using Clojure 1.2, use clojure-contrib 1.2, or the new modular Contrib libraries.
If you are using Clojure 1.3, use the new modular Contrib libraries.

As of the date of this reply I'm not sure if there is a version of clojure.contrib.prxml that is compatible with clojure-1.3 (Someone please correct me if I'm wrong).

So I would suggest using clojure-1.2.1 and clojure-contrib 1.2.0.

Clojure 1.2까지만 해도 상응하는 clojure-contrib 라이브러리가 있었습니다. 책은 예전 버전으로 씌여진 것이므로 제가 설치한 1.6 버전의 clojure에는 맞지 않는 구문이 있는 것 같습니다. 그래서 몇몇 함수들의 경우 책의 내용대로 실습해보면 이렇게 오류가 발생합니다.

D:\lisp>java -cp .\clojure-1.6.0\clojure-1.6.0\clojure-1.6.0.jar;.\clojure-contrib.jar clojure.main -i .\init_run.clj -r
Clojure 1.6.0

user=> (use '[clojure.contrib.lazy-seqs :only (primes)])
NoSuchMethodError clojure.lang.RestFn.<init>(I)V  clojure.contrib.def/defmacro---509 (def.clj:39)

그래도 clojure-contrib의 몇몇 함수는 정상 동작합니다.

user=> (str-join "te" "test")
"tteetestet"

가장 최신의 clojure.contrib를 보니,

Clojure Contrib Libraries 
; http://dev.clojure.org/display/doc/Clojure+Contrib+Libraries

"Modular Contrib"라고 명명되고 있으며 다음의 설명이 함께 하고 있습니다.

Modular Contrib projects (compatible with Clojure 1.2.0** thru 1.5.0, each in its own repo). **Mostly... support for Clojure 1.2.x is being dropped from several Contrib projects now - check the individual projects' test matrix on the build server to be sure of version support.


암튼, 저 같은 초보자에게는 좀 혼돈스럽습니다. ^^




그다음, 오탈자가 영문 서적 기준으로 다음의 사이트에서 제공됩니다.

http://pragprog.com/titles/shcloj/errata

한글 서적이 위의 것을 반영했는지는 확인해 보지 못했습니다. 참고로, 저도 오탈자를 하나 발견했는데요. "4장 117 페이지"에 보면,

take나 split 또는 drop으로 시작하는 함수는 모두 평가가 지연된 시퀀스(lazy sequence)를 반환한다.

라고 되어 있는데 직접 확인해 보면,

user=> (class (take-while (complement #{\a\e\i\o\u}) "the-quick-brown-fox"))
clojure.lang.LazySeq

이처럼 LazySeq로 나오는 것이 정상인데, split의 경우는 그것으로 시작한다고 모두 지연된 시퀀스는 아닙니다. split-at 함수가 그 한 예입니다.

user=> (class (split-at 5 (range 10)))
clojure.lang.PersistentVector




마지막으로, 이 책을 읽으면서 가장 이해가 안되었던 것이 recur 함수의 사용법이었습니다. 171페이지에 보면,

(defn parity [n]
  (loop [n n par 0]
    (if (= n 0)
      par
      (recur (dec n) (- 1 par)))))

라는 코드가 있는데 par의 값이 변하는 것인가?... 그렇다면 왜 그것이 변할 수 있도록 평가가 되는 것인지 이해가 안되었습니다. 예를 들어 다음의 코드는 par에 값이 저장되지 않습니다.

(defn parity [n]
  (let [par 0]
      (- 1 par) (println par)))

그래서 Clojure 포럼에 질문을 했는데 ^^

https://groups.google.com/forum/#!topic/lisp-korea/1waSJVuDORg

다음과 같은 친절한 설명을 들었습니다.

recur는 그 평가된 결과값이 다시 그 회귀시작점(recurtion point)으로 재바인딩됩니다.

http://clojure.org/special_forms#recur

위 코드에서는 회귀시작점(recursion point)가 loop이고, recur 문구(form)의 결과는 각각 loop의 바인딩 문구(form)의 n과 par에 재바인딩됩니다.

>(loop [i 10 j 5]
    (if (= j 0)
      [i j]
      (recur (inc i) (dec j))))
;=> [15 0]

recur의 회귀시작점(recursion point)는 loop 뿐 만 아니라 함수가 될 수도 있습니다.

> (defn f [i j]
     (if (= j 0)
       [i j]
       (recur (inc i) (dec j))))
> (f 10 5)

;=> [15 0]

그렇습니다. 회귀 시작점으로 재바인딩 된다는 저 사실을 모르는 상태에서는 parity 함수의 해석이 불가능했던 것입니다. 책에는 저렇게 친절하게 설명이 안되어 있습니다. ^^




암튼, LISP 언어를 이참에 접해봤는데... 역시나 어렵긴 합니다. 함수형 언어들이 최근 들어 재조명되고는 있지만, 글쎄요... 역시나 Clojure와 같은 언어들이 주류 언어에 편승하기에는 아직 시기상조인 듯 합니다. 달리 말하면, 아직 시간의 여유가 있으니 조금은 숨을 돌릴 수 있을 것 같다는! ^^




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







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

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

비밀번호

댓글 작성자
 



2024-05-17 08시33분
정성태

... 106  [107]  108  109  110  111  112  113  114  115  116  117  118  119  120  ...
NoWriterDateCnt.TitleFile(s)
11282정성태8/26/201720406Math: 22. 행렬로 바라보는 피보나치 수열
11281정성태8/26/201722611.NET Framework: 677. Visual Studio 2017 - NuGet 패키지를 직접 참조하는 PackageReference 지원 [2]
11280정성태8/24/201720084디버깅 기술: 94. windbg - 풀 덤프에 포함된 모든 모듈을 파일로 저장하는 방법
11279정성태8/23/201731409.NET Framework: 676. C# Thread가 Running 상태인지 아는 방법
11278정성태8/23/201719843오류 유형: 417. TFS - Warning - Unable to refresh ... because you have a pending edit. [1]
11277정성태8/23/201720999오류 유형: 416. msbuild - error MSB4062: The "TransformXml" task could not be loaded from the assembly
11276정성태8/23/201724853.NET Framework: 675. C# - (파일) 확장자와 연결된 실행 파일 경로 찾기 [2]파일 다운로드1
11275정성태8/23/201734217개발 환경 구성: 323. Visual Studio 설치 없이 빌드 환경 구성 - Visual Studio 2017용 Build Tools [1]
11274정성태8/22/201720863.NET Framework: 674. Thread 타입의 Suspend/Resume/Join 사용 관련 예외 처리
11273정성태8/22/201722336오류 유형: 415. 윈도우 업데이트 에러 Error 0x80070643
11272정성태8/21/201726186VS.NET IDE: 120. 비주얼 스튜디오 2017 버전 15.3.1 - C# 7.1 공개 [2]
11271정성태8/19/201720106VS.NET IDE: 119. Visual Studio 2017에서 .NET Core 2.0 프로젝트 환경 구성하는 방법
11270정성태8/17/201732295.NET Framework: 673. C#에서 enum을 boxing 없이 int로 변환하기 [2]
11269정성태8/17/201722398디버깅 기술: 93. windbg - 풀 덤프에서 .NET 스레드의 상태를 알아내는 방법
11268정성태8/14/201722455디버깅 기술: 92. windbg - C# Monitor Lock을 획득하고 있는 스레드 찾는 방법
11267정성태8/10/201726121.NET Framework: 672. 모노 개발 환경
11266정성태8/10/201726269.NET Framework: 671. C# 6.0 이상의 소스 코드를 Visual Studio 설치 없이 명령행에서 컴파일하는 방법
11265정성태8/10/201754173기타: 66. 도서: 시작하세요! C# 7.1 프로그래밍: 기본 문법부터 실전 예제까지 [11]
11264정성태8/9/201725560오류 유형: 414. UWP app을 signtool.exe로 서명 시 0x8007000b 오류 발생
11263정성태8/9/201720896오류 유형: 413. The C# project "..." is targeting ".NETFramework, Version=v4.0", which is not installed on this machine. [3]
11262정성태8/5/201719473오류 유형: 412. windbg - SOS does not support the current target architecture. [3]
11261정성태8/4/201721905디버깅 기술: 91. windbg - 풀 덤프 파일로부터 강력한 이름의 어셈블리 추출 후 사용하는 방법
11260정성태8/3/201720399.NET Framework: 670. C# - 실행 파일로부터 공개키를 추출하는 방법
11259정성태8/2/201718946.NET Framework: 669. 지연 서명된 어셈블리를 sn.exe -Vr 등록 없이 사용하는 방법
11258정성태8/1/201720271.NET Framework: 668. 지연 서명된 DLL과 서명된 DLL의 차이점파일 다운로드1
11257정성태7/31/201719894.NET Framework: 667. bypassTrustedAppStrongNames 옵션 설명파일 다운로드1
... 106  [107]  108  109  110  111  112  113  114  115  116  117  118  119  120  ...