- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我正在阅读 this question尝试深入了解答案。它专门询问传递引用,所有答案似乎都表明不支持传递引用。然而,this answer这意味着,虽然可能不支持通过引用传递,但某些值确实可以通过引用访问。一个更简单的例子是 cons cells;我可以将一个 cons 单元格传递给一个函数,然后将它的 cdr 或 car 更改为我喜欢的任何内容。
最后,我想知道(用 C# 的说法)值类型和引用类型之间是否有明确的区分,以及是否有任何方法(比上面引用的答案更方便)将值视为引用类型。
最佳答案
没有区别:所有对象在 Lisp 中都是按值传递的(至少在我所知道的所有 Lisp 中)。 但是一些对象是可变的,conses 就是这样一种类型。所以你可以将一个 cons cell 传递给一个过程并在该过程中改变它。因此,重要的考虑因素是对象是否可变。
特别是这个 (Common Lisp) 函数总是返回 T
作为它的第一个值,即使它的第二个值可能没有 0
作为它的汽车或 cdr。
(defun cbv (&optional (f #'identity))
(let ((c (cons 0 0)))
(let ((cc c))
(funcall f c)
(values (eq c cc) c))))
> (cbv (lambda (c)
(setf (car c) 1
(cdr c) 2)))
t
(1 . 2)
然而,由于 Common Lisp 具有词法作用域、一流的函数和宏,您可以做一些技巧,使它看起来有点像正在发生按引用调用:
(defmacro capture-binding (var)
;; Construct an object which captures a binding
`(lambda (&optional (new-val nil new-val-p))
(when new-val-p
(setf ,var new-val))
,var))
(defun captured-binding-value (cb)
;; value of a captured binding
(funcall cb))
(defun (setf captured-binding-value) (new cb)
;; change the value of a captured binding
(funcall cb new))
(defun cbd (&optional (f #'identity))
(let ((c (cons 0 0)))
(let ((cc c))
(funcall f (capture-binding c))
(values (eq c cc) c cc))))
现在:
> (cbd (lambda (b)
(setf (captured-binding-value b) 3)))
nil
3
(0 . 0)
如果您了解它是如何工作的,那么您可能会非常了解作用域和宏在 Lisp 中的工作方式。
在 Common Lisp 中按值传递对象的普遍性有一个异常(exception),Rainer 在下面的评论中提到了这一点:某些原始类型的实例可能会在某些情况下被复制以提高效率。这只会发生在特定类型的实例上,并且它发生的对象总是不可变的。为了处理这种情况,CL 提供了一个等式谓词 eql
这与 eq
做同样的事情, 除了它知道可能以这种方式 secret 复制的对象并正确地比较它们。
所以,安全的做法是使用 eql
而不是 eq
:因为可以复制的对象总是不可变的,这意味着你永远不会被这个。
这是一个示例,其中您自然认为相同的对象结果却不同。给定这个定义:
(defun compare (a b)
(values (eq a b)
(eql a b)))
然后在我使用的实现中我发现:
> (compare 1.0d0 1.0d0)
nil
t
所以 double 浮点零不是 eq
对它自己,总是,但它总是 eql
对它自己。并尝试一些看起来应该相同的东西:
> (let ((x 1.0d0)) (compare x x))
t
t
所以在这种情况下,看起来函数调用不是复制对象,而是我从读取器中的两个不同对象开始。然而,始终允许实现随意复制数字,并且很可能会使用不同的优化设置。
关于reference - 在Common Lisp中,什么时候引用对象,什么时候直接按值访问?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54018077/
最近,我开始学习 cuis-smalltalk,我没有意识到与 CLOS 相比,Smalltalk 的 OOP 有多么深刻和深入(我使用的是 Ruby)。我了解到 Smalltalk 是一个自己实现的
Maven存储库包含以下两个依赖项:org.apache.commons:commons-io:1.3.2和commons-io:commons-io:1.3.2。有什么区别,我应该在pom.xml中
我刚刚在我的 pom 文件中看到 Apache commons-collections 有两个不同的组 ID: commons-collections commons-collect
Windows 上的 Common Lisp 中是否有用于串行端口通信的库? 最佳答案 下面是一些使用 SBCL 外部函数 POSIX 调用实现串行通信的函数。它不如完整的库好,但我解决了根据此协议(
SBCL 64位,1.1.7 如果我想创建一个包并使用package:CL中的一些符号,我将创建一个像这样的包: (defpackage :foo (:import-from :cl
我正在忙着学习Common Lisp,并且正在寻找一种静态代码分析工具,该工具将帮助我开发更好的样式并避免陷入常见的陷阱。 我找到了Lisp Critic,看起来不错,但我希望有人可以推荐其他一些工具
我正在阅读《Practical Common Lisp》一书,在第 22 章第 284 页的脚注 5 中,我看到一段让我感到困惑的代码片段。 我知道变量list和tail有一个共同的列表结构,但我很困
我正在阅读 Practical Common Lisp ,并且对 Lisp 的 COPY-TREE 函数有疑问。 书中给出了调用的例子 (copy-tree '( '(1 2) '(3 4) '(5
我正在尝试使用 user guide 中的抓取示例运行 geb用于引入依赖项: $ cat my.groovy @Grapes([ @Grab("org.gebish:geb-core:0.9
这里一定有更好的方法,对吧? (format t "Enter your age: ~%") (defun age-case (age) (case age (1 (format t "Y
如何在 do 循环中绑定(bind)从函数返回的多个值? 以下显然是非常错误的,但是这样的事情可能吗? (do (((x y z) (3-val-fn) (3-val-fn))) ((equa
所以我正在学习 Lisp 做分数,这很棒。但是为什么这个相等性检查返回 NIL: * (= 0.2 1/5) NIL ...如果转换为 float 则返回 True第一的: * (=
是否可以“统计”一个文件并找到它的文件类型 - 常规或目录? 最佳答案 阅读关于 portable pathname library 的章节来自 Peter Seibel 的 Practical Co
我是 CL 的新手,正在使用 AllegroCL。我试图弄清楚如何组织我的源代码以满足以下要求: 我想阻止 src 代码包含我的测试套件。 我想以可移植的方式声明项目依赖项(src 和 test de
谁能告诉我最新的标准化 Common Lisp 的文档是什么(应该遵循各种实现的文档)?我问是因为我可以在网上找到很多关于 CL 的书都来自 90 年代,所以我想知道它们是否是最新的。我也来自于在 R
假设我必须定义一个名为foo 的函数。假设,为了定义它,我使用了一些辅助函数 foo1, foo2, foo3, ... 当我加载包含这些函数的文件时,我可以从顶层使用所有这些函数。相反,我只想从顶层
这拒绝编译。注释掉 (setf roll行让它编译。然而,(setf roll...本身在 REPL 中正确评估。 程序: ;; loop n times ; sum up number of hit
我目前正在学习 Common Lisp,并尝试将一些 JSON 发送到网络服务。我要发送的 JSON 以类似于以下的结构表示: ((:ITEMS ((:KEY . "value1") (:IGNO
我有一个带波浪号的目录名(作为字符串):~/projects . 我想得到它的完整路径:/home/user/projects .我怎么做 ? 目标是将它传递给 uiop:run-program ,这
我想从输入文件中读取一个字符串(用户可能修改也可能没有修改)。我想将此字符串视为使用固定数量的参数调用的格式指令。但是,我知道某些格式指令(特别是我想到的 ~/)可能会用于注入(inject)函数调用
我是一名优秀的程序员,十分优秀!