- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
学习不同的 Lisp 语言之间是否有任何协同作用?我目前正在学习 Emacs Lisp,因为它在我的日常 Emacs 使用中立即有用,但是我对所有 Lisp 都很着迷,所以也许有一天我会学习和使用其他语言。 当我开始深入研究 Common Lisp、Scheme 或 Clojure 时,学习 Emacs Lisp 对我有帮助吗?换句话说,这对我来说是学习一门全新的语言,还是一些概念和范式很常见?我也对比较 Lisp 之间的独特差异感兴趣,当我从一个 Lisp 转到另一个时,这将是一个问题。
在How to go about learning Common Lisp and Emacs Lisp?提到会有“浪费”,但没有详细说明,到什么程度。
最佳答案
如果您想学习一些较早的 Lisp 基础知识,那么 Emacs Lisp 就可以了。
Emacs Lisp:可以在 Emacs 中使用。包括开发环境。主流 Lisp 的旧方言。缺乏很多概念。有许多编辑器编程的扩展。 Common Lisp 和 Emacs Lisp 有一些直接的继承关系(命名、概念……)。
通用 Lisp。很多好书值得学习。许多 Lisp 概念(包括 OO 编程)可以通过 Common Lisp 学习。强烈推荐。 Common Lisp 内置了最重要的 Lisp 工具,其余的由库提供。可以从中学到很多东西。
方案:70 年代创建的不同 Lisp 方言。与 Emacs Lisp 或 Common Lisp 不直接兼容。许多优秀的书籍和其他教程 Material 。强烈推荐学习 Lisp 基础知识和一些更高级的东西。您对 Scheme 的研究越深入,它与 Emacs Lisp 甚至 Common Lisp 的区别就越大。
Clojure:非常不同的 Lisp 方言。与 Common Lisp、Emacs Lisp 或 Scheme 不兼容。共享一些概念,一些概念的工作方式不同。好书。如果您想学习一些 Lisp 或特别是 Clojure,推荐使用。 Clojure 强调函数式编程和并发编程 - 非常相关的主题。
如果你想学习更主流的 Lisp(具有典型 Lisp 方言的外观和感觉的 Lisp),我会推荐 Common Lisp 或 Scheme。
我对学习 Lisp (!) 的语言偏好是:
举个例子:
这是 McCarthy's Lisp 中的 COLLAPSE
函数,写于 1960 年(来自 Lisp I Programmer's manual, 1960,第 101 页)。它基本上就是许多 Lisp 练习中的 FLATTEN
函数。它采用嵌套列表并返回一个新列表,其中原子位于单个列表中。
DEFINE
(((COLLAPSE,(LAMBDA,(L),(COND,
((ATOM,L),(CONS,L,NIL))
((NULL,(CDR,L)),
(COND,((ATOM,(CAR,L)),L),(T,(COLLAPSE,(CAR,L)))))
(T,(APPEND,(COLLAPSE,(CAR,L)),(COLLAPSE,(CDR,L)))))
))))))
这是 Common Lisp 版本。您可以将其保留为大写或将其转换为小写。两者都有效。
(DEFUN COLLAPSE (L)
(COND
((ATOM L) (CONS L NIL))
((NULL (CDR L))
(COND ((ATOM (CAR L)) L)
(T (COLLAPSE (CAR L)))))
(T (APPEND (COLLAPSE (CAR L))
(COLLAPSE (CDR L))))))
基本上是一样的。只有定义函数的形式具有不同的名称和语法。否则代码完全相同。
试试 McCarthy 在 Common Lisp 中的例子:
CL-USER > (COLLAPSE '(((A B) ((C))) ((D (E F)) (G) ((H)))))
(A B C D E F G H)
它运行。
现在让我们使用 GNU Emacs 在 Emacs Lisp 中尝试一下。 Emacs Lisp 具有小写标识符:
ELISP> (defun collapse (l)
(cond
((atom l) (cons l nil))
((null (cdr l))
(cond ((atom (car l)) l)
(t (collapse (car l)))))
(t (append (collapse (car l))
(collapse (cdr l))))))
ELISP> (collapse '(((a b) ((c))) ((d (e f)) (g) ((h)))))
(a b c d e f g h)
它在 Emacs Lisp 中运行而无需更改。
您可以在 Scheme 中获得类似的版本(较小的重命名):
在 Petite Chez Scheme 中:
> (define collapse
(lambda (l)
(cond
((atom? l) (cons l '()))
((null? (cdr l))
(cond ((atom? (car l)) l)
(else (collapse (car l)))))
(else (append (collapse (car l))
(collapse (cdr l)))))))
我们可以使用DEFINE
来定义一个函数。 COND
看起来略有不同。 ()
是空列表。谓词添加了 ?
。
> (collapse '(((a b) ((c))) ((d (e f)) (g) ((h)))))
(a b c d e f g h)
运行。
在 Clojure 中它看起来会有所不同。基本上,您必须重新考虑大部分代码。
这是 Clojure 自己实现的flatten
:
(defn flatten
[x]
(filter (complement sequential?)
(rest (tree-seq sequential? seq x))))
您可以按照 Lisp 版本的精神编写一个 flatten
- 它看起来仍然会有所不同。
来自 rosetta.org :
(defn flatten [coll]
(lazy-seq
(when-let [s (seq coll)]
(if (coll? (first s))
(concat (flatten (first s)) (flatten (rest s)))
(cons (first s) (flatten (rest s)))))))
名称不同,语法不同,语义不同(适用于惰性序列而不是列表)。
Common Lisp、Emacs Lisp、Visual Lisp、ISLISP 等方言试图保留传统。
Scheme 或 Clojure 等方言不受名称和语法的束缚。他们在各个方面进行了创新。 Scheme 仍然提供旧功能的直接版本。 Clojure 没有。 Clojure 程序员不会将此视为劣势。
关于clojure - 学习一种 Lisp 有助于学习另一种吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11243936/
为什么该语言的名称是“Clojure”? 我用谷歌搜索了一下,在#clojure 中询问。到目前为止,还没有运气。 最佳答案 Rich Hickey(他是 Clojure 的设计者)对此的评论是 wi
我不明白为什么升级后会出现以下编译错误: Compiling addr-verify.core Exception in thread "main" java.lang.NoClassDefFound
我试图将从映射操作返回的(惰性)序列传递给另一个映射操作,以便我可以在第一个序列中查找元素。代码从文本文件(以行/列格式)解析一些足球装置,清理它,然后返回一张 map 。 这是代码: (ns fix
我想过滤一组,例如: (filter-set even? #{1 2 3 4 5}) ; => #{2 4} 如果我使用clojure.core/filter我得到一个不是集合的seq: (filte
(defn hi[](+ 5 6)) (hi) (defn hi[](+ 6 7)) (hi) 你好,我是 clojure 的新手。如上所述,我编写了两个具有相同名称的函数。我们可以在 cloj
我按照这个伪代码递归地将十进制转换为二进制。 findBinary(decimal) if (decimal == 0) binary = 0 else binar
我正在尝试学习 Clojure 并尝试定义这个简单的函数: user=> (defn triple [arg] (* 3 arg)) #'user/triple user=> (triple 1) 3
是->和 ->>宏只是为了使代码更具可读性还是它们还有其他特定功能? 最佳答案 线程优先( -> )和线程最后( ->> )是为了使代码更具可读性。但这已经很重要了! 它允许取消嵌套函数调用(示例取自
我在 http://www.learningclojure.com/2010/11/yet-another-way-to-write-factorial.html 上找到了这个代码,但我不明白 pop
我正在阅读 Programming Clojure 2nd edition,在第 49 页它涵盖了 Clojure 的 for 循环结构,它说它实际上是一个序列理解。 作者建议使用以下代码: (def
Clojure 中有双端队列吗?我的印象是 Clojure 的 PersistentQueue 是单端的(我错了吗?)。我需要能够从队列的任一端删除(即“pop”)和“peek”数据。我所说的双端队列
换句话说,有没有办法在看起来不像 (MACRO arg* ...) 的表单上触发宏扩展? . 举一个假设的例子: (defmacro my-var (do (printf "Using my-va
我很难理解懒惰。 有人能帮我理解为什么我下面的函数不是懒惰的吗 (defn my-red ([f coll] (my-red f (first coll) (rest coll) ))
在 Clojure 核心中决定参数函数顺序的规则是什么(如果有的话)? 类似 map 的函数和 filter期望数据结构作为最后一个 争论。 类似 assoc 的函数和 select-keys期待数据
我在 clojuredocs 上遇到过 completing 函数,但目前没有文档。 你能提供一些例子吗? 最佳答案 completing 用于扩充可能没有具有一元“完成”元数的一元重载的二元归约函数
这个现在支持吗?我能找到的唯一信息是来自维基的示例( https://github.com/clojure/core.match/wiki/Deftype-and-defrecord-matching
我正在关注“Clojure in Action”,对此我感到困惑: (defn with-log [function-to-call log-statement ] (fn [& args
对于下面的代码,箭头是宏还是函数名称中的简单字符? (来自 here) (defn file->map [file] ;; TODO ) 最佳答案 箭头是函数名称的一部分。有一个函数定义,不是
Clojure 的 range函数包含来自 start独家在end (如果提供)。核心库中是否有一个函数可以提供完全包含(开始和结束)的范围? 我发现在某些情况下必须调整最终值的代码 - 例如向下而不
当我尝试从 REPL 运行以下代码时(使用动态记录): (defrecord (symbol "rec2") (vec (map symbol ["f1" "f2"]))) 我收到错误 Compile
我是一名优秀的程序员,十分优秀!