CLisp 7:用LISP的基本规则实现if
2012-08-11 21:17
211 查看
LISP有7个基本规则,或7个基本函数,这里用他们实现if,在此过程中学一些知识
为了覆盖系统自带的if,我们将名称改成if.(后面加一个点号)
首先,定义一个名词为if.的函数
(defun if. (test success &optional fail)
(cond ((eq fail nil) (cond (test success)))
('t (cond (test success) ('t fail)))))
测试一下能不能工作
(if. t (print 'if) (print 'else))
if
else
if
输出的结果明显不正确
因为调用函数if.前,对输入的所有参数进行求值,即调用if.前就执行了print函数。
接着,把函数if.改成宏if.,避免对输入参数求值
(defmacro if. (test success &optional fail)
`(cond ((eq ,fail nil) (cond (,test ,success)))
('t (cond (,test ,success) ('t ,fail)))))
测试一下能不能工作
(if. t (print 'if) (print 'else))
else
if
if
输出的结果也不正确
问题出在(eq ,fail nil)上,宏展开后变成(eq (print 'else) nil),即执行了print函数。
要改变判断optional参数是否有效的方式
(defmacro if. (test success &optional (fail nil fail-p))
`(cond (fail-p (cond (,test ,success) ('t ,fail)))
('t (cond (,test ,success)))))
测试时出现错误 COND: variable FAIL-P has no value
不能用optional参数,改用函数重载,即定义两个if.宏
(defmacro if. (test success)
`(cond (,test ,success)))
(defmacro if. (test success fail)
`(cond (,test ,success) (t ,fail)))
到此为止能够通过测试,和系统自带的if有相同的输出
今天发现上面用option的实现存在问题,不应该在第一个cond表达式前面写反引号,这样做将判断有没有fail分支的代码带到了最终生成的代码中。正确的实现如下,这样不用写重载的两个if.宏
(defmacro if. (test success &optional fail)
(cond ((eq nil fail) `(cond (,test ,success)))
(t `(cond (,test ,success) (t ,fail)))))
为了覆盖系统自带的if,我们将名称改成if.(后面加一个点号)
首先,定义一个名词为if.的函数
(defun if. (test success &optional fail)
(cond ((eq fail nil) (cond (test success)))
('t (cond (test success) ('t fail)))))
测试一下能不能工作
(if. t (print 'if) (print 'else))
if
else
if
输出的结果明显不正确
因为调用函数if.前,对输入的所有参数进行求值,即调用if.前就执行了print函数。
接着,把函数if.改成宏if.,避免对输入参数求值
(defmacro if. (test success &optional fail)
`(cond ((eq ,fail nil) (cond (,test ,success)))
('t (cond (,test ,success) ('t ,fail)))))
测试一下能不能工作
(if. t (print 'if) (print 'else))
else
if
if
输出的结果也不正确
问题出在(eq ,fail nil)上,宏展开后变成(eq (print 'else) nil),即执行了print函数。
要改变判断optional参数是否有效的方式
(defmacro if. (test success &optional (fail nil fail-p))
`(cond (fail-p (cond (,test ,success) ('t ,fail)))
('t (cond (,test ,success)))))
测试时出现错误 COND: variable FAIL-P has no value
不能用optional参数,改用函数重载,即定义两个if.宏
(defmacro if. (test success)
`(cond (,test ,success)))
(defmacro if. (test success fail)
`(cond (,test ,success) (t ,fail)))
到此为止能够通过测试,和系统自带的if有相同的输出
今天发现上面用option的实现存在问题,不应该在第一个cond表达式前面写反引号,这样做将判断有没有fail分支的代码带到了最终生成的代码中。正确的实现如下,这样不用写重载的两个if.宏
(defmacro if. (test success &optional fail)
(cond ((eq nil fail) `(cond (,test ,success)))
(t `(cond (,test ,success) (t ,fail)))))
相关文章推荐
- CLisp 8:用LISP的基本规则实现while
- CLisp 10:用LISP的基本规则实现switch...case
- 集合的基本运算: 依据集合运算规则,实现任意给定两个集合的交、并、差、笛卡儿积运算,和第一个集合的幂集,并显示运算结果。
- 面向对象地分析Linux内核设备驱动(1):——Linux内核驱动中面向对象的基本规则和实现方法
- .net实现自定义UBB代码及规则的基本实现
- COM+架构的基本思想——实现IUnknown::QueryInterfacce方法时的规则
- Java中实现 Client-Server 体系1 -- 基本实现
- Java实现单向链表的基本功能详解
- 理解“从规则上实现”和“从技术上实现”的不同
- 使用c:if标签实现if else功能
- 设计一款好的移动App,有哪些基本规则?
- ArcGIS.Server.9.3和ArcGIS API for Flex实现基本的地图浏览(一)
- Android瀑布流照片墙实现,体验不规则排列的美感
- PHP 实现四种基本排序算法
- remove/remove_if算法配合容器的erase方法实现容器删除元素功能
- C# on rails! 基本信息CRUD操作的简单实现
- C++实现链表基本操作
- 写Java程序的三十个基本规则
- python 基本Kmeans算法实现
- C语言实现单链表的基本操作及其部分面试题