我承认我对 Lisp 的了解非常少。但是我对这门语言非常感兴趣,并计划在不久的将来开始认真学习它。我对这些问题的理解无疑是有缺陷的,所以如果我说了什么明显错误的话,请评论和纠正我,而不是投反对票。
真正的同音和自修改语言
我正在寻找支持 Homoiconicity(代码与数据具有相同的表示形式)和无限制的自我修改(无限制意味着您可以更改正在运行的代码的各个方面,而不仅仅是发出新代码或更改函数指针/代表。)
到目前为止,我发现只有三个例子符合这个标准:
- 机器码。Homoiconic 因为一切都是数字。可不受限制地修改,因为它包含指针,可用于操作任何内存地址,无论该地址是否包含代码或数据。
- 马尔堡。与机器代码相同的推理。每条指令在执行后都会自我修改
- 脱氧核糖核酸。不是编程语言,但仍然很有趣。它不像机器代码那样自我修改;实际指令+数据被修改到位的地方。然而,它是自我复制的,可以根据之前的状态变异/进化(辐射等副作用时不时地把它搞砸)。无论如何,这只是一种间接的自我修改方式。简而言之,DNA可以自我修改,但它是通过在其整体中复制自身以及相关突变来实现的。DNA 的物理串是“不可变的”。
为什么 Lisp 不在此列表中
Lisp 不在该列表中,因为在我看来 Lisp几乎是同音异形的,并且只支持受限的自我修改。你可以做类似的事情
(+ 1 2 3)
这将做同样的事情
(eval '(+ 1 2 3))
在第一个版本中(+ 1 2 3)
是原始代码,而在第二个版本中是数据。通过假设这个陈述的真实性,可以说 Lisp 甚至不是 homiconic。代码具有与数据相同的表示形式,因为它们都是列表/树/S 表达式。但是,您必须明确标记这些列表/树/S 表达式中的哪些是代码,哪些是对我来说是数据,这一事实似乎表明 Lisp 毕竟不是 homiconic。这些表示非常相似,但它们在微小的细节上有所不同,您必须实际说明您是在处理代码还是数据。这绝不是一件坏事(事实上其他任何事情都是疯狂的),但它突出了 Lisp 和机器代码之间的区别。在机器代码中,您不必明确标记哪些数字是指令,哪些是指针,哪些是数据。
这是反对不受限制的自我修改的更有力的案例。当然,您可以获取代表一些代码的列表并对其进行操作。例如改变
'(+ 1 2 3)
到
'(+ 1 4 3)
然后你运行它eval
。但是当你这样做时,你只是在编译一些代码并运行它。您不是在修改现有代码,您只是在发布和运行新代码。C# 可以使用表达式树做同样的事情,即使格式不太方便(这是由于 C# 代码对其 AST 具有不同的表示形式,而不是 Lisp,它是它自己的 AST)。您是否真的可以获取整个源文件并在整个源文件运行时开始修改它,对源文件所做的更改会对程序行为产生实时影响?
除非有某种方法可以做到这一点,否则 Lisp 既不是 homiconic 也不是自我修改的。(为了推迟对定义的争论,Lisp 不像机器代码那样同音或自我修改。)
使 Lisp Homoiconic/无限制地自我修改的方法
我可以看到 3 种潜在的方法来使 Lisp 像机器代码一样具有同义性/可自我修改。
- 非冯诺依曼架构。如果有人可以发明一些惊人的假设机器,其中程序的最低级别表示是可以直接执行的 AST(无需进一步编译)。在这样的机器上,AST 既代表可执行指令,也代表数据。不幸的是,问题还没有解决,因为 AST 仍然必须是代码或数据。eval 函数的出现不会改变这一点。在机器代码中,您可以根据需要在代码和数据之间来回切换。而使用 eval 和 Lisp 后,一旦您将某个列表从数据“评估”到代码并执行了它,就无法再次将该列表作为数据取回。实际上,该列表已永远消失,并已被其价值所取代。我们会遗漏一些关键的东西,这恰好是指针。
- 列出标签。如果要求每个列表也具有唯一标签,则可以通过对具有给定标签的列表运行函数来进行间接自我修改。结合延续,这最终将允许在与机器代码相同的意义上自我修改代码。标签等同于机器代码内存地址。例如,考虑一个 Lisp 程序,其中 AST 的顶部节点具有标签“main”。然后,在 main 中,您可以执行一个函数,该函数接受一个标签、一个整数、一个原子,并将原子复制到列表中,其标签与提供给函数的标签相匹配,位于整数指定的索引处。然后只需在 main 上调用当前的继续。你去,自我修改代码。
- Lisp 宏。我没有花时间去理解 Lisp 宏,它们实际上可能完全符合我的想法。
第 1 点与第 2 点相结合将产生一个完全自我修改的 Lisp。前提是可以生产所描述的神奇的 Lisp 机器。2. 单独可以产生一个自我修改的 Lisp,但是在冯诺依曼架构上的实现可能非常低效。
问题
- 除了机器码、dna 和 malbolge 之外,还有什么语言可以进行完全的自我修改并且是同音的?
- (如果您在上面的文字中做了一个 tl;dr,请不要费心回答)。lisp 真的是谐音+自我修改吗?如果你这么说,你能准确引用我的论点中我误入歧途的地方吗?
附录
具有不受限制的自我修改但没有同音性的语言
- 集会。该代码使用单词而不是数字,因此失去了同义性,但它仍然具有指针,它保留了对内存的完全控制并允许不受限制的自我修改。
- 任何使用原始指针的语言。例如 C/C++/Objective C。与 Assembly 相同的参数
- 包含虚拟指针的 JIT 语言。例如在不安全的上下文中运行的 C#/.net。与大会相同的论点。
其他可能相关/有趣的概念和语言:Lisp、Ruby、Snobol、Forth 和它的编译时元编程、Smalltalk 和它的反射、无类型的 lambda 演算及其属性,即一切都是函数(这意味着假设我们可以发明一台直接执行 lambda 演算的机器,lambda 演算将是同形的,而 Von Neumann 机器代码在该机器上运行时不会。[并且 Godels theorem 将是可执行的。哈哈,可怕的想法:P])