Microsoft MVP성태의 닷넷 이야기
개발 환경 구성: 241. 책 "프로그래밍 클로저: Lisp"을 읽고 나서. [링크 복사], [링크+제목 복사],
조회: 23363
글쓴 사람
정성태 (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분
정성태

[1]  2  3  4  5  6  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13945정성태6/7/2025356오류 유형: 960. 파이썬 + conda - mysqlclient 사용 시 "NameError: name '_mysql' is not defined" 에러
13944정성태6/7/2025349오류 유형: 959. The trust relationship between this workstation and the primary domain failed. - 네 번째 이야기
13943정성태6/6/2025491개발 환경 구성: 748. Windows + Foundry Local - 로컬에서 AI 모델 활용
13942정성태6/5/2025685오류 유형: 958. winget 설치 시 "0x80d02002 : unknown error"
13941정성태6/2/2025920닷넷: 2334. C# - cpuid 명령어를 이용한 CPU 제조사 문자열 가져오기파일 다운로드1
13940정성태6/1/20251333C/C++: 188. C++의 32비트 + Release 어셈블리 코드를 .NET으로 포팅할 때 주의할 점파일 다운로드1
13939정성태5/29/20251612오류 유형: 957. NVIDIA Triton Inference Server - version `GLIBCXX_3.4.32' not found (required by /opt/tritonserver/backends/python/triton_python_backend_stub)
13938정성태5/29/20251366개발 환경 구성: 747. 파이썬 - WSL/docker에 구성한 Triton 예제 개발 환경
13937정성태5/24/20251346개발 환경 구성: 746. Windows + WSL2 환경에서 (tensorflow 등의) NVIDIA GPU 인식
13936정성태5/23/20251175개발 환경 구성: 745. Linux / WSL 환경에 Miniconda 설치하기
13935정성태5/20/20251193파이썬 - pip 사용 시 "ImportError: cannot import name 'html5lib' from 'pip._vendor'" 오류
13934정성태5/20/20251703스크립트: 77. 파이썬 - 'urllib.request' 모듈의 명시적/암시적 로딩 차이
13933정성태5/19/20251272오류 유형: 956. Visual Studio 2022가 17.12 버전부터 업데이트 되지 않는다면?
13932정성태5/18/20251477스크립트: 76. 파이썬 - Version 문자열 다루기(semver 패키지)
13931정성태5/17/20251767스크립트: 75. 파이썬 - Cython 기본 예제 및 컴파일
13930정성태5/17/20251445개발 환경 구성: 744. 파이썬 - Windows embeddable package 환경에서 외부 패키지 사용하는 방법(ex: UFO² 환경 구성)
13929정성태5/16/20251476오류 유형: 955. 파이썬 - "Windows embeddable package" REPL 환경에서 "NameError: name 'exit' is not defined"
13928정성태5/15/20251525오류 유형: 954. UFO² - "'Invalid URL (POST /v1/chat/completions/chat/completions)'"
13927정성태5/15/20251522오류 유형: 953. OpenAI - The API request of HOST_AGENT failed: OpenAI API request exceeded rate limit: Error code: 429
13926정성태5/14/20251900개발 환경 구성: 743. LLM과 윈도우의 만남 - Desktop AgentOS UFO² 기본 환경 구성
13925정성태5/12/20252000닷넷: 2333. C# - (Console 유형의 프로젝트에서) Clipboard 연동파일 다운로드1
13924정성태5/8/20251736닷넷: 2332. C# - (JetBrains Omea Reader 대상으로) 런타임 시에 메서드 가로채기 [2]파일 다운로드1
13923정성태5/5/20251490스크립트: 74. 파이썬 - C# - Python.NET의 RunSimpleScript, Exec, Eval 차이점파일 다운로드1
13922정성태5/3/20251727스크립트: 73. 파이썬 - Windows embeddable package 버전에서 tkinter 환경 구성
13921정성태5/3/20252226오류 유형: 952. 듀얼 채널 메모리 정렬을 지키지 않은 컴퓨터의 Windows 비정상 종료 현상(Blue Screen) [2]
[1]  2  3  4  5  6  7  8  9  10  11  12  13  14  15  ...