2017-01-16 201 views

回答

8

有兩個原因來命名匿名函數(或者,我已經至少有兩個原因這樣做)。首先是給它一個名字告訴後面的讀者(也許你自己在6個月後),匿名函數應該做什麼。

第二個是(正如你所提到的)在堆棧跟蹤中有更好的信息,以便在發生故障時將代碼指向代碼中的正確位置。函數被編譯成類,並且類名包括函數名稱的(消失)版本。當你有一個堆棧跟蹤時,它將包含該類名稱,並因此指出你的語義朝向正確的位置。

user=> (filter (fn [x] (/ 100 x)) [100 50 0]) 
ArithmeticException Divide by zero clojure.lang.Numbers.divide (Numbers.java:158) 
user=> (pst *e) 
ArithmeticException Divide by zero 
    clojure.lang.Numbers.divide (Numbers.java:158) 
    clojure.lang.Numbers.divide (Numbers.java:3784) 
    user/eval8/fn--9 (NO_SOURCE_FILE:3) 
    clojure.core/filter/fn--6908 (core.clj:2790) 
    ... 
nil 

user=> (filter (fn hundred-div [x] (/ 100 x)) [100 50 0]) 
ArithmeticException Divide by zero clojure.lang.Numbers.divide (Numbers.java:158) 
user=> (pst *e) 
ArithmeticException Divide by zero 
    clojure.lang.Numbers.divide (Numbers.java:158) 
    clojure.lang.Numbers.divide (Numbers.java:3784) 
    user/eval14/hundred-div--15 (NO_SOURCE_FILE:5)  ;; <--- 
    clojure.core/filter/fn--6908 (core.clj:2790) 
    ... 
5

命名匿名函數可能是有用的,並且也通過被印有他們的名字:

user=> ((fn [] (throw (Exception. "unnamed")))) 

Exception unnamed user/eval805/fn--806 (NO_SOURCE_FILE:1) 
user=> ((fn myfn [] (throw (Exception. "named")))) 

Exception named user/eval809/myfn--810 (NO_SOURCE_FILE:1) 
7

除了是蹤跡有用,我想,當你需要一個匿名函數是遞歸的,你可以使用它,因爲這將能夠調用自身。

例如:

(fn factorial[n] 
    (if (<= n 1) 
    1 
    (* n (factorial (- n 1))))) 

雖然遞歸這樣在Clojure是有點危險,因爲這有可能導致堆棧溢出。