Coursera课程 Programming Languages, Part B 总结
碎言碎语
- 很多没有写过 Lisp 程序的人都会对 Lisp 中的括号产生偏见(包括曾经第一次看到 Lisp 程序的我),事实上,括号赋予 Lisp 的是严谨的程序结构,Lisp 中的括号是极其严格的,往往既缺一不可,也不能画蛇添足。括号往往意味着表达式的计算,例如
(e)表示 e 表达式的计算,那么一旦 e 是一个确定的值,则会出现问题,(3),在 Racket 中会报错:application: not a procedure;。 - 这门课最值得一提的就是 week 2 的作业,在 Racket 语言中实现自己的语言 MUPL(Made Up Programming Language),Racket 作为解释器去执行用自己的语言写成的代码。当然话是这么说,其实介绍作业的材料已经解决了从 0 到 1 的困难了,从定义语言的基本语法,到扩展这门语言,再到用这门语言编写函数,都有着详细的指导。
- Racket 作为解释器,如果要执行我们的语言,简单情况下,只需要编写一个 eval 函数,里面充满着各种判断语句来解析我们的语言(语法)。另外,我们可以在自己的语言中定义宏的写法,其实这里宏可以就是一个 Racket 函数,这个函数会返回一个 MUPL 表达式,在 eval 函数执行的时候可以很自然的替换,这不是很类似 Racket 语言中的宏吗?检测宏是否运行正常只要看它生成的代码是否正确就行了。
- 从类型检查谈起,并讨论了弱类型语言以及强类型语言的区别(如 C 的数组越界),并在某些方面比较了动态类型和静态类型的优缺点。在 Racket 中我们仅仅使用 list 就定义出很多数据结构,通过 car cdr 可以很容易的使用(如最基本的树,也可以实现 set 和 map),但是定义一个新的类型,既可以更快地帮助类型测试捕捉到这个错误,也提高了代码的可读性。
- 如果对 Part B 的内容还意犹未尽的话,还有 SICP 可以食用,前三章会看的很轻松(或许)。
笔记
1. 惰性求值(Lazy Evaluation)
- call-by-value:在调用函数时,表达式先被求值再作为参数。
- call-by-need:在需要的时候再计算。
在大多数编程语言中都是第一种方式,在一些语言中我们可以用一种既巧妙又简单的方式实现第二种方式,即惰性求值(Lazy Evaluation)。
下面举得例子并不是非常恰当,但是很直观。
Racket 中的 if 表达式是这样的:
(if e1 e2 e3)
也就是如果 e1 为真,执行 e2 否则执行 e3。很显然它接受三个参数,那么有一个问题,如果应用的是第一种方式,e1 e2 e3 先被计算再判断 e1 真假的话,一旦 e2 表达式产生了 side effect (例如打印函数),则会导致程序的不理想输出,甚至于有时会发生无限递归。
为了解决这个问题,e2 和 e3 在判断 e1 之前不能被计算。
在 Racket 语言中,我们直接在 e2 和 e3 外面分别套一个 lambda表达式 即可达到这个目的。
(define (bad-if e1 e2 e3)
(if e1 e2 e3))
(bad-if (= 1 2) (write 1) (write 3)) ; 13
(define (new-if e1 e2 e3)
(if e1 (e2) (e3)))
(new-if (= 1 2) (lambda () (write 1)) (lambda () (write 3))) ; 3
所以,如果我们想要延迟求值,把要延迟求值的表达式放到一个 lambda表达式 中即可,在需要用到时对函数求值即可。
这样问题是解决了,但是显得异常麻烦,有几个参数就要写几个 lambda表达式,有没有什么简洁的方法呢?
这个时候就要用到 宏(macro) 了。
宏的替换发生在任何表达式的计算之前。
(define-syntax good-if
(syntax-rules ()
[(new-if e1 e2 e3)
(new-if e1 (lambda () e2) (lambda () e3))]))
(define (new-if e1 e2 e3)
(if e1 (e2) (e3)))
(good-if (= 1 2) (write 1) (write 3)) ; 3
虽然这个例子毫无意义,但是思想还是能体现不少的。
惰性求值一个常见的应用就是 流(stream) 了。
具体可参考 SICP 。
2. macro
模拟 for 循环。
(define-syntax for
(syntax-rules (to do)
[(for lo to hi do body)
(let ([l lo]
[h hi])
(letrec ([loop (lambda (it)
(cond [(= it h) body]
[(< it h) (begin body (loop (+ it 1)))]))])
(loop l)))]))
(for 1 to 5 do (write "H"))
Coursera课程 Programming Languages, Part B 总结的更多相关文章
- Coursera课程 Programming Languages, Part A 总结
Coursera CSE341: Programming Languages 感谢华盛顿大学 Dan Grossman 老师 以及 Coursera . 碎言碎语 这只是 Programming La ...
- Coursera课程 Programming Languages 总结
课程 Programming Languages, Part A Programming Languages, Part B Programming Languages, Part C CSE341: ...
- Coursera课程 Programming Languages, Part C 总结
碎言碎语 和前面的 ML 和 Racket 感觉明显不一样了,一边学着一边觉得这真是一门奇怪的语言,有着各种奇怪的语法,不过真的算是一个奇妙的体验(相比前面的两门语言,Ruby 的学习资源多了不少). ...
- 【Python学习笔记】Coursera课程《Using Python to Access Web Data》 密歇根大学 Charles Severance——Week6 JSON and the REST Architecture课堂笔记
Coursera课程<Using Python to Access Web Data> 密歇根大学 Week6 JSON and the REST Architecture 13.5 Ja ...
- 【C语言】Coursera课程《计算机程式设计》台湾大学刘邦锋——Week6 String课堂笔记
Coursera课程 <计算机程式设计>台湾大学 刘邦锋 Week6 String 6-1 Character and ASCII 字符变量的声明 char c; C语言使用一个位元组来储 ...
- ESSENTIALS OF PROGRAMMING LANGUAGES (THIRD EDITION) :编程语言的本质 —— (一)
# Foreword> # 序 This book brings you face-to-face with the most fundamental idea in computer prog ...
- Coursera课程下载和存档计划[转载]
上周三收到Coursera平台的群发邮件,大意是Coursera将在6月30号彻底关闭旧的课程平台,全面升级到新的课程平台上,一些旧的课程资源(课程视频.课程资料)将不再保存,如果你之前学习过相关的课 ...
- Natural language style method declaration and usages in programming languages
More descriptive way to declare and use a method in programming languages At present, in most progra ...
- The future of programming languages
In this video from JAOO Aarhus 2008 Anders Hejlsberg takes a look at the future of programming langu ...
随机推荐
- ABP 框架启程
准备动手写一套电商的系统,辗转收集了不少相关的开源项目,最后决定使用ABP作为起点. 在园子里好多人都在推广ABP.有个园友做了一个集合贴,方便大家使用 ABP集合贴 建议大家优先看 HK Zhan ...
- java读取请求中body数据
java读取请求中body数据 /** * 获取request中body数据 * * @author lifq * * 2017年2月24日 下午2:29:06 * @throws IOExcepti ...
- Ubuntu 下 libgps 库的使用
简介 一般 GPS 接收器会遵循美国国家海洋电子协会(National Marine Electronics Association)所指定的标准规格,其中包含传输资料的格式以及传输资料的通讯协议.那 ...
- ZKWeb网页框架2.0正式发布
2.0.0更新的内容有 更新框架要求 框架要求从 netstandard 1.6 升到 netstandard 2.0 框架要求从 netcoreapp1.1 升到 netcoreapp2.0 更新引 ...
- 使用MyBatis缓存
(1).为什么需要使用缓存:: MyBatis是一个持久层(数据库层)映射框架,在所有访问数据库的操作中,无疑数据查询是最耗费数据库资源的操作了,因为你一次可能需要查询成千上百万条记录(如果你不加限制 ...
- Axure学习笔记(一)
Axture是一种快速制作原型的工具,在产品经理和体验设计师之中非常流行,不过现在产品经理比较难找,所以我只好上阵研究了一下. 经过几天的研究,看了小楼老师的一些视频,看了一些文档,做了 ...
- 《调试九法——软硬件错误的排查之道》【PDF】下载
<调试九法--软硬件错误的排查之道>[PDF]下载链接: https://u253469.ctfile.com/fs/253469-231196352 内容简介 <调试九法:软硬件错 ...
- 《决战大数据:驾驭未来商业的利器》【PDF】下载
内容简介 大数据时代的来临,给当今的商业带来了极大的冲击,多数电商人无不"谈大数据色变",并呈现出一种观望.迷茫.手足无措的状态.车品觉,作为一名经验丰富的电商人,在敬畏大数据的同 ...
- LR接口压力测试实战(限量抢红包接口)
一.业务描述:微信群中,运营人员放出活动链接,用户单击活动链接进入活动页面如下图,用户输入手机号抢红包(红包数量有限) 二.接口测试代码 Action() { lr_rendezvous(" ...
- Python学习日记:day5-------dict字典
#字典dict------->唯一的映射类型 1.数据类型的划分 数据类型划分为可变数据类型和不可变数据类型. 不可变数据类型:tupe(元组).bool.int.str 可 ...