如何在clojure中定义安全的sqrt函数?

Tim*_*Tim 3 clojure genetic-programming

我正在使用fungp(遗传编程工具)来模拟复杂的函数并且遇到sqrt问题.

基本上,我必须将函数及其arity的向量传递给fungp,以便它可以组合它们的表达式.然后将评估表达式并返回最佳表达式.这个函数向量看起来像:

(def functions
  '[[+ 2]
  [- 2]
  [* 2]
  [fungp.util/abs 1]
  [fungp.util/sdiv 2]
  [fungp.util/sin 1]
  [fungp.util/sqrt 1]
  [inc 1]
  [dec 1]])
Run Code Online (Sandbox Code Playgroud)

这个设置给了我一百行错误,如:

#<ClassCastException java.lang.ClassCastException: java.lang.Double cannot be cast to clojure.lang.IFn>
Run Code Online (Sandbox Code Playgroud)

我认为这是由于fungp.util/sqrt的定义:

(defn sqrt [x] (if (x > 0) (Math/sqrt x) 0))
Run Code Online (Sandbox Code Playgroud)

我认为0导致评估失败,但我不确定.我已经尝试定义我自己的安全平方根版本,但无法正确获取语法.

所以,这就是我被困住的地方.我需要一个安全的平方根版本(负输入返回0)并在fungp表达式中正确评估.

编辑:为了完整性,这是我尝试编写自己的平方根包装器的(很多)变体之一:

(defn sqrt-fn [x] `(if (~x > 0) (Math/sqrt ~x) 0))
Run Code Online (Sandbox Code Playgroud)

输出(中间位是从函数生成的表达式):

#<ClassCastException java.lang.ClassCastException: clojure.lang.Cons cannot be cast to java.lang.Number>

(let [] (- (dec (- (- (fungp.util/sin (tutorial.tut1/sqrt-fn 8.0)) (fungp.util/sdiv (* x 2.0) (dec 9.0))) (fungp.util/sdiv (tutorial.tut1/sqrt-fn (* x x)) (- (- x 4.0) (+ x x))))) (fungp.util/sdiv (tutorial.tut1/sqrt-fn (fungp.util/sin (
+ (dec x) (inc x)))) (fungp.util/sdiv (* (inc (inc 1.0)) (* (+ x 9.0) (fungp.util/sin 9.0))) (tutorial.tut1/sqrt-fn (- (tutorial.tut1/sqrt-fn x) (fungp.util/abs 3.0)))))))

NullPointerException   clojure.lang.Numbers.ops (Numbers.java:942)
Run Code Online (Sandbox Code Playgroud)

我不写表达式,所以如果有额外的括号或缺少括号,它们来自我定义sqrt的方式.

Ale*_*lex 6

这有几个问题:

(defn sqrt-fn [x] `(if (~x > 0) (Math/sqrt ~x) 0))
Run Code Online (Sandbox Code Playgroud)

首先,正如评论中所暗示的那样,(x > 0)试图将x(可能是一个数字)称为函数.>是功能,所以它必须首先,如在(> x 0).

此外,您可能不希望它们的语法引用 - 这会阻止对内容的评估,因此您的函数返回引用的符号列表.