SICP CONCLUSION

让我们举起杯,祝福那些将他们的思想镶嵌在重重括号之间的Lisp程序员 !

祝我能够突破层层代码,找到住在里计算机的神灵!

目录

1. 构造过程抽象
2. 构造数据抽象
3. 模块化、对象和状态
4. 元语言抽象
5. 寄存器机器里的计算

Chapter 3

  • 模块化、对象和状态
练习答案

从前两章中我们认识到,在克服系统复杂性问题时,构造过程抽象和数据抽象祈祷了非常关键的作用。但是在进行组织模块化后,我们还需要一些组织原则来完成系统的整体设计,使这些系统自然地划分为内聚的部分

有一种强有力的设计策略,就是基于外界事物的结构,去模拟设计,也就是划分每个计算对象,也就是面向对象思想,这一策略的好处也就是在扩展修改增加程序时,只需要做一些局部的工作

另外一种策略是将注意力集中在流过系统的信息流上

赋值和局部状态

作为一个对象,我们就可以算它的状态是受时间或者历史影响的,那么就需要表示它的状态变量,而所谓的交互就是建立对象与对象之间状态变量的联系,形成内部联系紧密,而与其他子系统只存在着松散的联系

  • 银行账户例子
balance 100
;: (withdraw 25)
75
;: (withdraw 25)
50
;: (withdraw 60)
Not enough
;: (withdraw 15)
35
(define (make-account balance)
(define (withdraw amount)
(if (>= balance amount)
(begin (set! balance (- balance amount))
balance)
"Insufficient funds"))
(define (deposit amount)
(set! balance (+ balance amount))
balance)
(define (dispatch m)
(cond ((eq? m 'withdraw) withdraw)
((eq? m 'deposit) deposit)
(else (error "Unknown request -- MAKE-ACCOUNT"
m))))
dispatch) ;: (define acc (make-account 100)) ;: ((acc 'withdraw) 50)
;: ((acc 'withdraw) 60)
;: ((acc 'deposit) 40)
;: ((acc 'withdraw) 60)
  • 引进赋值带来的利益

蒙特卡洛模拟例子

6/Π2是随机选取两个整数之间没有公共因子的概率

使用了随机数生成器:内部拥有状态变量

(define (estimate-pi trials)
(sqrt (/ 6 (monte-carlo trials cesaro-test)))) (define (cesaro-test)
(= (gcd (rand) (rand)) 1)) (define (monte-carlo trials experiment)
(define (iter trials-remaining trials-passed)
(cond ((= trials-remaining 0)
(/ trials-passed trials))
((experiment)
(iter (- trials-remaining 1) (+ trials-passed 1)))
(else
(iter (- trials-remaining 1) trials-passed))))
(iter trials 0))

不使用随机数生成器

(define (estimate-pi trials)
(sqrt (/ 6 (random-gcd-test trials random-init)))) (define (random-gcd-test trials initial-x)
(define (iter trials-remaining trials-passed x)
(let ((x1 (rand-update x)))
(let ((x2 (rand-update x1)))
(cond ((= trials-remaining 0)
(/ trials-passed trials))
((= (gcd x1 x2) 1)
(iter (- trials-remaining 1)
(+ trials-passed 1)
x2))
(else
(iter (- trials-remaining 1)
trials-passed
x2))))))
(iter trials 0 initial-x))

虽然现在还没有看出非常大的问题,但是其中已经破坏了程序的模块性,而且暴露本应该是内部的状态变量。而之前的程序反应出的是一个复杂的计算性过程,其他部分都像在随着时间不断变化,并且隐藏起随时间变化的内部,而这就需要局部变量去模拟系统的状态,并用赋值来模拟它们的变化。增强模块性

  • 引进赋值的代价

引进了赋值可以去模拟系统的变化,但是这也迫使我们需要引进新的计算模型,因为代换模型已经不适用了。如果我们不使用赋值,以同样参数对同一过程的两次求值可以产生同样的的结果,这样的就称为函数式程序设计

(define (make-decrementer balance) ;; 函数式编程,所以两次调用并不能改变状态
(lambda (amount)
(- balance amount)))
  • 同一和变化

一旦我们将变化引进了我们的计算模型,首先考虑两个物体实际上同一的概念

;: (define D1 (make-decrementer 25))
;: (define D2 (make-decrementer 25))

如果我们说D1 D2是同一的是可接受的,因为调用它并不会改变其内部状态

;: (define W1 (make-simplified-withdraw 25))
;: (define W2 (make-simplified-withdraw 25))

但是W1 W2很显然就不是同一的了

如果一个语言支持在表达式里“同一的东西可以替换”,那么久称这个语言是具有引用透明性的

我们只能通过改变一个对象,去观察另一个对象是否发生变化,以此来判断这两个是不是同一的,但如果不能通过观察对象两次,看看一次观察中看到的某些对象性质是否与另一次不同,我们又怎么能清楚一个对象是否变化了呢?所以如果没有有关同一的某些先验观念,我们也就不能确定变化,而不能看到变化久不能确定同一性

1)
;: (define peter-acc (make-account 100))
;: (define paul-acc (make-account 100))
;:
2)
;: (define peter-acc (make-account 100))
;: (define paul-acc peter-acc)

在第一种情况下,很显然这两个是不同对象,但在2中,修改一个对象也就会修改另外一个对象,所以在构造计算模型的时候,就很容易引起混乱,比如在面向对象中有关对象的传递,同一或许有点让人迷惑。但是如果我们保证绝不修改数据对象,那么有关同一的概念就又不同了,就可以将一个数据对象完全看作是由其片段组成的了。在有理数中可以看作它是由分子分母组成的,所以如果修改了它的分字或者分母,它就不在是一个同一对象了。但是对于银行账户,如果你改变了它的账户,它依旧是同一对象。

  • 命令式函数设计的缺陷

在命令式函数中广泛使用赋值,这就会引进一个复杂的问题,就是赋值的顺序,状态有关时间的变化

这一节主要是在需要一种更好的组织系统的设计方式后,一种是面向对象的方式,一种是流的方式,在基于面向对象的基础上引入了局部变量和赋值来描述计算模型的状态,这样的好处的是使程序更加的具有模块化,在每个模块中都有自己的局部变量去描述自身的状态,但这其中也有代价,就是发生了同一概念的复杂性

SICP读书笔记 3.1的更多相关文章

  1. 【SICP读书笔记(一)】正则序展开的特殊情况

    scheme解释器有两种实现方式,一种是应用序,先对每个参数求值,再以首过程对所有求得的参数求值. 第二种是正则序,会“完全展开然后归约”(书中原文) SICP中的练习1.5,让我困惑了一下.原题如下 ...

  2. SICP读书笔记 1.1

    SICP CONCLUSION 让我们举起杯,祝福那些将他们的思想镶嵌在重重括号之间的Lisp程序员 ! 祝我能够突破层层代码,找到住在里计算机的神灵! 目录 1. 构造过程抽象 2. 构造数据抽象 ...

  3. SICP读书笔记 3.5

    SICP CONCLUSION 让我们举起杯,祝福那些将他们的思想镶嵌在重重括号之间的Lisp程序员 ! 祝我能够突破层层代码,找到住在里计算机的神灵! 目录 1. 构造过程抽象 2. 构造数据抽象 ...

  4. SICP读书笔记 3.4

    SICP CONCLUSION 让我们举起杯,祝福那些将他们的思想镶嵌在重重括号之间的Lisp程序员 ! 祝我能够突破层层代码,找到住在里计算机的神灵! 目录 1. 构造过程抽象 2. 构造数据抽象 ...

  5. SICP读书笔记 3.2

    SICP CONCLUSION 让我们举起杯,祝福那些将他们的思想镶嵌在重重括号之间的Lisp程序员 ! 祝我能够突破层层代码,找到住在里计算机的神灵! 目录 1. 构造过程抽象 2. 构造数据抽象 ...

  6. SICP读书笔记 3.3

    SICP CONCLUSION 让我们举起杯,祝福那些将他们的思想镶嵌在重重括号之间的Lisp程序员 ! 祝我能够突破层层代码,找到住在里计算机的神灵! 目录 1. 构造过程抽象 2. 构造数据抽象 ...

  7. SICP读书笔记 2.5

    SICP CONCLUSION 让我们举起杯,祝福那些将他们的思想镶嵌在重重括号之间的Lisp程序员 ! 祝我能够突破层层代码,找到住在里计算机的神灵! 目录 1. 构造过程抽象 2. 构造数据抽象 ...

  8. SICP读书笔记 2.4

    SICP CONCLUSION 让我们举起杯,祝福那些将他们的思想镶嵌在重重括号之间的Lisp程序员 ! 祝我能够突破层层代码,找到住在里计算机的神灵! 目录 1. 构造过程抽象 2. 构造数据抽象 ...

  9. SICP读书笔记 2.3

    SICP CONCLUSION 让我们举起杯,祝福那些将他们的思想镶嵌在重重括号之间的Lisp程序员 ! 祝我能够突破层层代码,找到住在里计算机的神灵! 目录 1. 构造过程抽象 2. 构造数据抽象 ...

随机推荐

  1. 1090. [SCOI2003]字符串折叠【区间DP】

    Description 折叠的定义如下: 1. 一个字符串可以看成它自身的折叠.记作S  S 2. X(S)是X(X>1)个S连接在一起的串的折叠.记作X(S)  SSSS…S(X个S). ...

  2. openfalcon-0.2 配置

    hbs 配置文件 { "debug": true, "database": "openfalcon:123456@tcp(172.16.230.188 ...

  3. C++网络爬虫的实现——WinSock编程

    写了一个网络爬虫,可以抓取网上的图片. 需要给定初始网站即可. 在vs2010中编译通过. 需要使用多字节字符集进行编译, vs2010默认的是Unicode字符集. 编译后,运行即可,有惊喜哦!!! ...

  4. DBlink的创建与删除

    创建方式一: create [public] database link link名称 connect to 对方数据库用户identified by 对方数据库用户密码 using  '对方数据库i ...

  5. 电商 APP 下单页(俗称车2) 业务流程概要设计

    购物车是电商APP的一个关键功能点,一般购物车包含 3-4 个页面,分别是: 1.购物车的商品列表页 2.商品下单页 3.订单付款页面 4.订单付款成功页面 由于现有购物车逻辑相对混乱,这里重新整理一 ...

  6. “C++动态绑定”相关问题探讨

    一.相关问题: 1. 基类.派生类的构造和析构顺序 2. 基类.派生类中virtual的取舍 二.测试代码: #include <iostream> class A { public: A ...

  7. c++类模板之友元函数

    前言:自从开始学模板了后,小编在练习的过程中.常常一编译之后出现几十个错误,而且还是那种看都看不懂那种(此刻只想一句MMP).于是写了便写了类模板友元函数的用法这篇博客.来记录一下自己的学习. 普通友 ...

  8. flex 自适应

    flex-grow.flex-shrink.flex-basis这三个属性的作用是:在flex布局中,父元素在不同宽度下,子元素是如何分配父元素的空间的. 其中,这三个属性都是在子元素上设置的. 注: ...

  9. 嵌入式C语言自我修养 02:Linux 内核驱动中的指定初始化

    2.1 什么是指定初始化 在标准 C 中,当我们定义并初始化一个数组时,常用方法如下: ] = {,,,,,,,,}; 按照这种固定的顺序,我们可以依次给 a[0] 和 a[8] 赋值.因为没有对 a ...

  10. Verilog 位拼接运算符的优先级

    最近研究FIFO的时候,在开源工程中看到这样一段代码 ; always @(posedge rd_clk) {'b0}}; else {'b0}}; else if(re) rp_bin <= ...