- Java锁的逻辑(结合对象头和ObjectMonitor)
- 还在用饼状图?来瞧瞧这些炫酷的百分比可视化新图形(附代码实现)⛵
- 自动注册实体类到EntityFrameworkCore上下文,并适配ABP及ABPVNext
- 基于Sklearn机器学习代码实战
即使在变化中,它也丝毫未变.
——赫拉克利特 。
吾犹昔人,非昔人也.
——僧肇 。
前面我们介绍了组成程序的各种基本元素,看到了如何把基本过程和基本数据组合起来,构造出复合的实体。不过对于设计程序而言,这些手段还不够,我们还需要一些能够帮助我们构造起 模块化 (modular)的大型系统的策略。所谓模块化,也即使这些系统能够“自然地”划分为一些 内聚 (coherent)的部分,使这些部分可以分别进行开发和维护.
在哲学上,组织程序的方式与我们对被模拟系统的认识息息相关。接下来我们要研究两种特色很鲜明的组织策略,它们源自于对于系统结构的两种非常不同的“ 世界观 ”(world views).
第一种策略将注意力集中在 对象 (objects)上,将一个大型系统看成不同对象的集合,它们的状态和行为可能随着时间不断变化.
另一种组织策略将注意力集中在流过系统的 信息流 (streams of information)上,非常像EE工程师观察一个信号处理系统.
这两种策略都对程序设计提出了具有重要意义的语言要求。对于对象途径而言,我们必须关注对象可以怎样变化而又保持其标识(identity)。这将迫使我们抛弃前面说讲过的计算的代换模型,转向更机械式的,理论上也更不容易把握的计算的 环境模型 (environment model)。在处理对象、变化和标识时,各种困难的根源在于我们需要在这一计算模型中与时间搏斗,如果引入并发后还将变得更糟糕。流方式将我们的模型中的模拟时间与求值过程中的事件发生顺序进行解耦,我们将通过一种称为 延时求值 (lazy evaluation)的技术做到这一点.
本节我们先介绍第一种对象世界观.
在对象世界观里,我们想让计算对象具有随着时间变化的状态,而这就需要让每个计算对象有自己的一些局部状态变量。现在让我们来对一个银行账户支取现金的情况做一个模拟。我们将用一个过程 withdraw 完成此事,它有一个参数 amount 表示支取的现金量。如果余额足够则 withdraw 返回支取之后账户里剩余的款额,否则返回消息 Insufficient funds (金额不足)。假设开始时账户有100元钱,在不断使用 withdraw 的过程中我们可能得到下面的响应序列:
withdraw(25) # 70
withdraw(25) # 50
withdraw(60) # "In sufficient funds"
withdraw(15) # 35
在这里可以看到表达式 widthdraw(25) 求值了两次,但它产生的值却不同,这是过程的一种新的行为方式。之前我们看到的过程都可以看做是一些可计算的数学函数的描述,两次调用一个同一个过程,总会产生出相同的结果.
为了实现 withdraw ,我们可以用一个全局变量 balance 表示账户里的现金金额,并将 withdraw 定义为一个访问 balance 的过程。下面是 balance 和 widthdraw 的定义:
balance = 100
def withdraw(amount):
global balance
if balance > amount:
balance = balance - amount
return balance
else:
return "Insufficient funds"
虽然 withdraw 能像我们期望的那样工作,变量 balance 却表现出一个问题。如上所示, balance 是定义在全局环境中的一个名字,因此可以被任何过程检查或修改。我们希望将 balance 做成 withdraw 内部的东西,因为这将使 withdraw 成为唯一能直接访问 balance 的过程,而其他过程只能间接地(通过对 withdraw 的调用)访问 balance 。这样才能准确地模拟有关的概念:balance是一个只有 withdraw 使用的 局部状态变量 ,用于保存账户状态的变化轨迹.
我们可以通过下面的方式重写出 withdraw ,使 balance 成为它内部的东西:
def new_withdraw():
balance = 100
def inner(amount):
nonlocal balance
if balance > amount:
balance = balance - amount
return balance
else:
return "Insufficient funds"
return inner
W = new_withdraw()
print(W(25)) # 70
print(W(25)) # 50
print(W(60)) # "In sufficient funds"
print(W(15)) # 35
这里的做法是用创建起一个包含局部变量 balance 的环境,并使它初始值为100。在这个环境里,我们创建了一个过程 inner ,它以 amount 作为一个参数,其行为就像是前面的 withdraw 过程。这样最终返回的过程就是 new_withdraw ,它的行为方式就像是 withdraw ,但其中的变量确实其他任何过程都不能访问的。用程序设计语言的行话,我们说变量 balance 被称为是 封装 在 new_withedraw 过程里面.
将赋值语句与局部变量相结合,形成了一种具有一般性的程序设计技术,我们将一直使用这种技术区构造带有局部状态的计算对象。但这一技术也带来了麻烦,我们之前在代换模型中说,应用(apply)一个过程应该解释为在将过程的形式参数用对应的值取代之后再求值这一过程。但现在出现了麻烦,一旦在语言中引进了赋值,代换就不再适合作为过程应用的模型了(我们将在3.1.3节中看到其中的原因)。我们需要为过程应用开发一个新模型,这一模型将在3.2节中介绍。现在我们要首先检查 new_withdraw 所提出的问题的几种变形.
下面过程 make_withdraw 能创建出一种“提款处理器”。 make_withdraw 的形式参数 balance 描述了有关账户的初始余额值.
def make_withdraw(balance):
def withdraw(amount):
nonlocal balance
if balance > amount:
balance = balance - amount
return balance
else:
return "Insufficient funds"
return withdraw
下面用 make_withdraw 创建了两个对象:
W1 = make_withdraw(100)
W2 = make_withdraw(100)
print(W1(50)) # 50
print(W2(70)) # 30
print(W2(40)) # Insufficient funds
print(W1(40)) # 10
我们可以看到, W1 和 W2 是相互完全独立的对象,每一个都有自己的局部状态变量 balance ,从一个对象提款与另一个毫无关系.
我们还可以创建出除了提款还能够存入款项的对象,这样就可以表示简单的银行账户了。下面是一个过程,它返回一个具有给点初始余额的“银行账户对象”:
def make_account(balance):
def withdraw(amount):
nonlocal balance
if balance >= amount:
balance = balance - amount
return balance
else:
return "In sufficient funds"
def deposit(amount):
nonlocal balance
balance = balance + amount
return balance
def dispatch(m):
nonlocal balance
if m == "withdraw":
return withdraw
if m == "deposit":
return deposit
else:
raise ValueError("Unkown request -- MAKE_ACOUNT %s" % m)
return dispatch
对于 make_acount 的每次调用将设置好一个带有局部状态变量 balance 的环境,在这个环境里, make_account 定义了能够访问 balance 过程 deposit 和 withdraw ,另外还有一个过程 dispatch ,它以一个“消息”做为输入,返回这两个局部过程之一。过程 dispatch 本身将会被返回,做为表示有关银行账户对象的值。这正好是我们在2.4.3节中看到过的程序设计的 消息传递 风格,当然这里将它与修改局部变量的功能一起使用.
acc = make_account(100)
print(acc("withdraw")(50)) # 50
print(acc("withdraw")(60)) # In sufficient funds
print(acc("deposit")(40)) # 90
print(acc("withdraw")(60)) # 30
对 acc 的每次调用将返回局部定义的 deposit 或者 withdraw 过程,这个过程随后被应用于给定的 amount 。就像 make_withdraw 一样,对 make_amount 的另一次调用 。
acc2 = make_acount(100)
将产生出另一个完全独立的账户对象,维持着它自己的局部 balance .
这里再举一个实现 累加器 的例子(事实上该例子在《黑客与画家》 [2] 第13章中也有出现,被用来说明不同编程语言编程能力的差异)。累加器是一个过程,反复用数值参数调用它,就会使得它的各个参数累加到一个和中。每次调用时累加器将返回当前的累加和。请写出一个生成累加器的过程 make_accumulator ,它所生成的每个累加器维持着一个独立的和。传给 make_accumulator 的输入描述了和的初始值。其Python实现代码如下:
def make_accumulator(sum_value):
def accumulator(number):
nonlocal sum_value
sum_value += number
return sum_value
return accumulator
A = make_accumulator(5)
print(A(10)) # 15
print(A(10)) # 25
当然,Common Lisp的写法将更为简单:
(defun make_accumulator (sum_value)
(lambda (number) (incf sum_value number)))
Ruby的写法与Lisp几乎完全相同:
def make_accumulator (sum_value)
lambda {|number| sum_value += number } end
《黑客与画家》中作者还展示了Perl、Smalltalk、JavaScript等语言的写法,感兴趣的朋友可以去看下这本书(剧透一下:作者这儿把Java黑惨了233).
正如下面将要看到的,将赋值引进所用的程序设计语言中,将会使我们陷入困难概念问题的丛林之中。但无论如何,将系统看做是带有局部状态的对象的集合,也是一种维护模块化设计的强有力技术。先让我们看一个简单的例子:如何设计出一个过程 rand ,每次它被调用就会返回一个随机选出的整数。这里的“随机选择”的意思并不清楚,其实我们希望的就是对 rand 的反复调用将产生一个具有均匀分布统计性质的序列。假定我们已经有一个过程 rand-update ,它的性质就是,如果从一个给点的数 x1 开始,执行下面操作 。
x2 = random_update(x1)
x3 = random_update(x2)
得到的值序列 x1 、 x2 , x3 ,...将具有我们所希望的性质.
实现 random_update 的一种常见方法就是采用将 \(x\) 更新为 \(ax+b\) 取模 \(m\) 的规则,其中 a 、 b 和 m 都是适当选出的整数。比如:
def rand_update(x):
a = int(pow(7, 5))
b = 0
m = int(pow(2, 31)) - 1
return (a * x + b) % m
Knuth的TAOCP第二卷(半数值算法) [3] 中包含了有关随机数序列和建立起统计性质的深入讨论。注意, random_update 是计算一个数学函数,两次给它同一个输入,它将产生同一个输出。这样,如果“随机”强调的事序列中每个数与前面的数无关的话,由 random_update 生成的数序列肯定不是“随机的”。在“真正的随机性”与所谓 伪随机 序列(由定义良好的确定性计算产生出的但又具有适当统计性质的序列)之间的关系是一个非常复杂的问题,涉及到数学和哲学中的一些困难问题,Kolmogorov、Solomonoff、Chaitin为这些问题做出了很多贡献,从Chaitin 1975 [4] 可以找到有关讨论.
现在回到当前的话题来。我们已经实现好了 random_update ,接下来在此基础上实现 rand 。我们可以将 rand 实现为一个带有局部状态变量 x 的过程,其中将这个变量初始化为某个固定值 rand_init 。对 rand 的每次调用算出当前 \(x\) 值的 random_update 值:
def make_rand(random_init):
x = random_init
def inner():
nonlocal x
x = rand_update(x)
return x
return inner
rand = make_rand(42)
print(rand()) # 705894
print(rand()) # 1126542223
当然,即使不用赋值,我们也可以通过简单地调用 rand_update ,生成同样的随机序列。但是这意味着程序中任何使用随机数的部分都必须显式地记住,需要将 x 的当前值传给 rand_update 作为参数,这样会徒增烦恼.
接下来,我们考虑用随机数实现一种称为 蒙特卡罗模拟 的技术.
蒙特卡罗方法包括从一个大集合里随机选择试验样本,并在对这些试验结果的统计估计的基础上做出推断。例如, \(6/\pi^2\) 是随机选取的两个整数之间没有公共因子(也即最大公因子为1)的概率。我们可以利用这一事实做出 \(\pi\) 的近似值(这个定理出自Cesaro,见TAOCP第二卷 [3] 4.5.2的讨论和证明).
这一程序的核心是过程 monte_carlo ,它以某个试验的次数( trails )以及这个试验本身( experiment )作为参数。试验用一个无参过程 cesaro_test 表示,返回的是每次运行的结果为真或假。 monte_carlo 运行指定次数的这个试验,它返回所做的这些试验中得到真的比例.
rand = make_rand(42)
import math
def estimate_pi(trials):
return math.sqrt(6 / monte_carlo(trials, cesaro_test))
def cesaro_test():
return math.gcd(rand(), rand()) == 1
def monte_carlo(trials, experiment):
def iter(trials_remaining, trials_passed):
if trials_remaining == 0:
return trials_passed / trials
elif cesaro_test():
return iter(trials_remaining - 1, trials_passed + 1)
else:
return iter(trials_remaining - 1, trials_passed)
return iter(trials, 0)
print(estimate_pi(500)) # 3.178208630818641
现在让我们试一试不用 rand ,直接用 rand_update 完成同一个计算。如果我们不使用赋值去模拟局部状态,那么将不得不采取下面的做法:
random_init = 42
def estimate_pi(trials):
return math.sqrt(6 / random_gcd_test(trials, random_init))
def random_gcd_test(trials, initial_x):
def iter(trials_remaining, trials_passed, x):
x1 = rand_update(x)
x2 = rand_update(x1)
if trials_remaining == 0:
return trials_passed / trials
elif math.gcd(x1, x2) == 1:
return iter(trials_remaining - 1, trials_passed + 1, x2)
else:
return iter(trials_remaining - 1, trials_passed, x2)
return iter(trials, 0, initial_x)
print(estimate_pi(500)) # 3.178208630818641
虽然这个程序还是比较简单的,但它却在模块化上打开了一些令人痛苦的缺口,因为它需要显式地去操作随机数 x1 和 x2 ,并通过一个迭代过程将 x2 传给 random_update 作为新的输入。这种对于随机数的显式处理与积累检查结果的结构交织在一起。此外,就连上层的过程 estimate_pi 也必须关心提供随机数的问题。由于内部的随机数生成器被暴露了出来,进入了程序的其它部分,我们很难将蒙特卡罗方法的思想隔离出来了。反观我们在程序的第一个版本中,由于通过赋值将随机数生成器的状态隔离在过程 rand 的内部,因此就使随机数生成的细节完全独立于程序的其它部分了.
由上面的蒙特卡洛方法实例体现的一种普遍性系统设计原则就是: 对于行为随时间变化的计算对象(如银行账户和随机数生成器),我们需要设置局部状态变量,并用对这些变量的赋值去模拟状态的变化 .
正如上面所看到的,赋值操作使我们可以模拟带有局部状态的对象。然而,这一获益也有一个代价,也即使我们的程序设计语言不能再用前面所提到过的代换模型解释了。进一步说,任何具有“漂亮”数学性质的简单模型,都不可能继续适合作为处理程序设计语言里的对象和赋值的框架了.
只要我们不适用赋值,以同样参数对同一过程的两次求值一定产生出同样的结果,因此就可以认为过程是在计算数学函数。就像我们在之前的章节中所提到的那样,不用任何复制的程序设计称为 函数式程序设计 .
要理解复制将怎样使事情复杂化了,考虑3.1.1节中 make_withdraw 过程的一个简化版本,其中不再关注是否有足够余额的问题:
def make_simplified_withdraw(balance):
def simplified_withdraw(amount):
nonlocal balance
balance = balance - amount
return balance
return simplified_withdraw
W = make_simplified_withdraw(25)
print(W(20)) # 5
print(W(10)) # -5
请将这一过程与下面 make_decrementer 过程做一个比较,该过程里没有用赋值运算:
def make_decrementer(balance):
return lambda amount: balance - amount
make_decrementer 返回的是一个过程,该过程从指定的量 balance 中减去其输入,但顺序调用时却不会像 make_simplifed_withdraw 那样产生累积的结果.
D = make_decrementer(25)
print(D(20)) # 5
print(D(10)) # 15
我们可以用代换模型解释 make_decrementer 如何工作。例如,让我们分析一下下面表达式的求值过程:
make_decrementer(25)(20)
首先简化组合式中的操作符,用25代换 make_decrementer 体里的 balance ,这样就规约出了下面的表达式:
(lambda amount: 25 - amount) (20)
随后应用运算符,用 20 代换 lambda 表达体里的 amount :
25 - 20
最后结果是5.
现在再来看看,如果将类似的代换分析用于 make_simplifed_withdraw ,会出现什么情况:
make_simplified_withdraw(25)(20)
先简化其中的运算符,用 25 代换 make_simplified_withdraw 体里的 balance ,这样就规约出了下面的表达式(注意,Python的lambda表达式里不能进行赋值运算(据Guido说是故意加以限制从而防止Python成为一门函数式编程语言),下面这个式子不能在Python解释器中运行,只是为了方便大家理解):
(lambda amount: balance = 25 - amount)(25)(20)
这里我们没有代换赋值表达式里的 balance ,因为赋值符号 = 的左边部分并不会进行求值,如果代换掉它,得到的 25 = 25 - amount 根本就没有意义.
现在用 20 代换 lambda 表达式体里的 amount :
(balance = 25 - 20)(25)
如果我们坚持使用代换模型,那么就必须说,这个过程应用的结果是首先将 balance 设置为5,而后返回25作为表达式的值。这样得到的结果当然是错误的。为了得到正确答案,我们不得不对 balance 的第一次出现(在 = 作用之前)和它的第二次出现(在 = 作用之后)加以区分,而代换模型根本无法完成这件事情.
这里的麻烦在于,从本质上说代换的最终基础就是, 这一语言里的符号不过是作为值的名字 。而一旦引入了赋值运算 = 和变量的值可以变化的想法,一个变量就不再是一个简单的名字了。现在的一个变量索引着一个可以保存值的位置(place),而存储再那里的值也是可以改变的。在3.2节里将会看到,在我们的计算模型里,环境将怎样扮演者“位置”的角色.
同一和变化 。
这里暴露出的问题远远不是简单地打破了一个特定计算模型,它还使得以前非常简单明了的概念现在都变得有问题了。首先考虑两个物体实际上“同一”(“the same”)的概念.
假定我们用同样的参数调用 make_decrementer 两次,就会创建出两个过程:
D1 = make_decrementer(25)
D2 = make_decrementer(25)
D1 和 D2 是同一的吗?“是”是一个可接受的回答,因为 D1 和 D2 具有同样的计算行为——都是同样的将会从其输入里减去25点过程。事实上,我们确实可以在任何计算中用 D1 代替 D2 而不会改变结果,如下所示:
print(D1(20)) # 5
print(D1(20)) # 5
print(D2(20)) # 5
于此相对应的是调用 make_simplified_withdraw 两次:
W1 = make_simplified_withdraw(25)
W2 = make_simplified_withdraw(25)
W1 和 W2 是同一的吗?显然不是,因为对 W1 和 W2 的调用会有不同的效果,下面的调用显示出这方面的情况:
print(W1(20)) # 5
print(W1(20)) # -15
print(W2(20)) # 5
虽然 W1 和 W2 都是通过对同样表达式 make_simplified_withdraw(25) 求值创建起来的东西,从这个角度可以说它们“同一”。但如果说在任何表达式里都可以用 W1 代替 W2 ,而不会改变表达式的求值结果,那就不对了.
如果一个语言支持在表达式里“同一的东西可以相互替换”的观念,这样替换不会改变有关表达式的值,这个语言就称为是具有 引用透明性 。而当我们的计算机语言包含赋值运算之后,就打破了引用透明性.
一旦我们抛弃了引用透明性,有关计算对象“同一”的意义问题就很难形式地定义清楚了。事实上,在我们企图用计算机程序去模拟的现实世界里,“同一”的意义本身就很难搞清楚的,这是由于“同一”和“变化”的 循环定义 所致:我们想要确定两个看起来同一的事物是否确实是“同一个东西”,我们一般只能去改变其中一个对象,看另一个对象是否也同样改变;但如果不观察“同一个”对象两次,看看对象的性质是否与另一次不同,我们就能确定对象是否“变化”。由是观之,我们必须要将“同一”作为一个先验观念引入(PS:这里可以参见康德的思想),否则我们就不可能确定“变化”.
现在举例说明这一问题会如何出现在程序设计里。现在考虑一种新情况,假定Peter和Paul有银行账户,其中有100块钱。关于这一事实的如下模拟:
peter_acc = make_account(100)
paul_acc = make_account(100)
和如下模拟之间有着实质性的不同:
peter_acc = make_account(100)
paul_acc = peter_acc
在前一种情况里,有关的两个银行账户互不相同。Peter所做的交易将不会影响Paul的账户,反之亦然。比如,当Peter取10块,Paul取10块,则Paul账户里还有90块:
peter_acc("withdraw")(10)
print(paul_acc("withdraw")(10)) # 90
而对于后一种情况,这里把 paul_acc 定义为与 peter_acc 是同一个东西,结果就使现在Peter和Paul共有一个共同的账户,此时当Peter取10块钱,Paul再取10块钱后,Paul就只剩80块钱了:
peter_acc("withdraw")(10)
print(paul_acc("withdraw")(10)) # 80
这里一个计算对象可以通过多于一个名字访问的现象称为 别名 (aliasing)。这里的银行账户例子是最简单的,我们在3.3节里还将看到一些更复杂的例子,例如“不同”的数据结构共享某些部分,如果对某一个对象的修改可能由于“副作用”而修改了另一“不同的”的对象,因为这两个“不同”对象实际上只是同一个对象的不同别名,当我们忘记这一情况程序就可能出现错误。这种错误被称为 副作用错误 ,特别难以定位和分析。因此某些人(如分布式计算大佬Lampson)就建议说,程序设计语言的设计不允许副作用或者别名 .
命令式程序设计的缺陷 。
与函数式程序设计相对应的,广泛采用赋值的程序设计被称为 命令式程序设计(imperative programming) 。除了会导致计算模型的复杂性之外,以命令式风格写出的程序还容易出现一些不会在函数式程序中出现的错误。举例来说,现在重看一下在1.2.1节里的迭代求阶乘程序:
def factorial(n):
def iter(product, counter):
if counter > n:
return product
else:
return iter(counter * product, counter + 1)
return iter(1, 1)
print(factorial(4)) # 24
我们也可以不通过内部迭代循环(这里假设Python支持尾递归)传递参数,而是采用更命令的风格,显式地通过赋值去更新变量 product 和 counter 的值:
def factorial(n):
product, counter = 1, 1
def iter():
nonlocal product, counter
if counter > n:
return product
else:
product = counter * product
counter = counter + 1
return iter()
return iter()
print(factorial(4)) # 24
这样做不会改变程序的结果,但却会引进一个很微妙的陷阱。我们应该如何确定两个赋值的顺序呢?像上面的程序虽然是正确的,但如果以相反的顺序写出这两个赋值:
counter = counter + 1
product = counter * product
就会产生出与上面不同的错误结果:
print(factorial(4)) # 120, Wrong!
一般而言,带有赋值的程序将强迫人们去考虑赋值的相对顺序,以保证每个语句所用的是被修改变量的正确版本。在函数式程序设计中,这类问题根本就不会出现 。事实上这种看法也说明,大部分的引论性程序设计课程采用高度命令式风格讲授,这确实是意见令人啼笑皆非的事情。这一情况可能源自20世纪70年代中流行的一种常见看法的残存遗迹,这种看法说调用过程的程序一定比执行赋值的程序效率更低(Steele(1977) [6] 批驳了这一论断)。还有,这种情况也可能反应了另一种观点,认为初学者一步步的看赋值比观察过程调用更容易。无论出于什么原因,它总是给初学程序设计的人们增加了关注“我应该把这个变量的赋值放在另一个之前呢还是之后”的负担,这会使程序设计复杂化,也使其中的主要思想变模糊了.
如果考虑有多个并发执行的进程的应用程序,命令式程序设计的复杂性还会变得更糟糕。我们将在3.4节回到这个问题.
最后此篇关于SICP:赋值和局部状态(Python实现)的文章就讲到这里了,如果你想了解更多关于SICP:赋值和局部状态(Python实现)的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我目前正在研究 SICP 关于逻辑编程的部分,但我被困在有关逻辑推导的示例中,尤其是附加到表单规则。它们是如何工作的?我不太明白的是第二条规则是如何取消第一个列表的。例如,给定: (规则(附加到表单(
据我了解,著名的SICP讲座视频: http://ocw.mit.edu/courses/electrical-engineering-and-computer-science/6-001-struc
我已经以“边做边学”的方式编程了将近 2 年,我认为自己相当不错,但是,我真的希望为计算机科学/计算机工程打下良好的基础,大多数人建议我开始关闭 SICP。 (计算机程序的结构和解释) 我想知道 这是
我正在研究计算机程序的结构和解释。马上,有两个词被使用,似乎表示一些特定的概念:计算过程和程序。我有一个简单的问题。在 Abelson 和 Sussman 的用法上下文中,这些术语的定义是什么? 最佳
求值器完整实现代码我已经上传到了GitHub仓库: TinySCM ,感兴趣的童鞋可以前往查看。这里顺便强烈推荐UC Berkeley的同名课程 CS 61A 。 在这个层次结构
即使在变化中,它也丝毫未变。 ——赫拉克利特 吾犹昔人,非昔人也。 ——僧肇 前面我们介绍了组成程序的各种基本元素,看到了如何把基本过程和基本数据组合
绪论 我们在第一章引进复合过程时,采用了求值的代换模型定义了将过程应用于实参(arguments)的意义: 将一个复合过程应用于一些实参,也就意味着用实参替换过程体里对应的形参(f
绪论 我们已经介绍过数据抽象,这是一种构造系统的方法学,它能够使程序中的大部分描述与其所操作的数据对象的具体表示无关,比如一个有理数程序的设计与有理数的实现相分离。这里的关键是构筑数据抽象屏障
我在这里问了几个关于 Scheme/SICP 的问题,答案经常涉及使用 apply 过程,我在 SICP 中没有看到过,在本书的索引中,它只列出了一次,结果是脚注。 一些使用示例基本上是这个问题的每个
我目前正在通过计算机程序的结构和解释来完成本书和 Brian Harvey(他有时很搞笑)的讲座,但是我还没有真正拥有我的“啊哈!时刻”区分函数和程序。 现在,我在讲座和阅读之外进行了研究,发现了一些
关于 SICP 3.5 我自己的实现如下 (define (delay exp) (lambda () exp)) (define (force delayed-obj) (delayed-obj
考虑以下定义: (define foo (lambda (x y) (if (= x y) 0 (+ x (foo (+
我正在使用 SICP 书并且做了这个练习: 1.11 函数 f 由以下规则定义:如果 n 3. 编写一个通过递归过程计算 f 的过程。编写一个通过迭代过程计算 f 的过程。 递归过程很简单。迭代的更难
我在阅读 SICP 时遇到了一个问题,在第 1 章中有一个名为 counting change 的示例,我需要在 scheme 中编写一个程序来计算给定一半的任何给定数字的可能变化方式- 美元、25
我正在使用 SICP 这本书,我正在为递归与迭代过程的概念而苦苦挣扎。在问题 1.17 他们问这个: 练习 1.17。本节中的取幂算法基于通过重复乘法执行取幂。类似地,可以通过重复加法来进行整数乘法。
在关于不动点的第 1 章中,书上说我们可以使用以下方法找到某些函数的不动点 f(x) = f(f(x)) = f(f(f(x))) .... 那些功能是什么? 当我将它重写为 y = y/2 时它对
我正在学习 SICP 这本很棒的书。尽管很棒,但它确实是一本难懂的书。我遇到了长尾递归问题,id est,迭代过程的递归定义。 这本书介绍了阶乘的迭代过程: (define (factorial n)
我正在使用 SICP 这本书并尝试解决这个练习: 1.2.4 Exponentiation Exercise 1.18. Using the results of exercises 1.16 and
这是生成递归过程的 SICP 的阶乘过程。 (define (factorial n) (if (= n 1) 1 (* n (factorial (- n 1)))))
这个问题在这里已经有了答案: SICP recursive process vs iterative process: using a recursive procedure to generate
我是一名优秀的程序员,十分优秀!