• 博客园logo
  • 会员
  • 众包
  • 新闻
  • 博问
  • 闪存
  • 赞助商
  • HarmonyOS
  • Chat2DB
    • 搜索
      所有博客
    • 搜索
      当前博客
  • 写随笔 我的博客 短消息 简洁模式
    用户头像
    我的博客 我的园子 账号设置 会员中心 简洁模式 ... 退出登录
    注册 登录
String Of Brilliant Blue
湛蓝之弦
博客园    首页    新随笔    联系   管理     

对scheme的一些理解(3)

断断续续读到SICP第三章,觉得scheme有点入门了,不过长久不练习,脑子又不能适应函数式编程模式了,觉得3.17、3.18、3.19还比较有意思,贴个自己的解题思路吧。

断断续续读到SICP第三章,觉得scheme有点入门了,不过长久不练习,脑子又不能适应函数式编程模式了,觉得3.17、3.18、3.19还比较有意思,贴个自己的解题思路吧。

3.17

(define (count-unique-pair x)
  (let ((db (cons 0 0)))
    (define (count-pair x)
      (define (add-unique-pair x db1)
 (define (add-record db0 x)
   (set-car! db0 x)
   (set-cdr! db0 (cons 0 0))
   x)
 (define (find-record x db0)
   (if (not (pair? (car db0)))
       (add-record db0 x)
       (if (eq? x (car db0))
    '()
    (find-record x (cdr db0)))))
 (find-record x db1))
      (if (not (pair? x))
   0
   (if (null? (add-unique-pair x db))
       0
       (+ (count-pair (car x))
   (count-pair (cdr x))
   1))))
    (count-pair x)))

(define xx '(a b))
(define z1 (cons xx xx))
(define z2 (cons '(a b) '(a b)))
(count-unique-pair z1)

实现比较丑陋,基本还是过程式编程,使用了一个db存储已经遍历过的pair,可以对付带环的表,而且只要此pair已经遍历过,则其子pair也不在遍历了。相比一下纯函数式写法,这种方式还是比较清晰且容易理解的,不过存在内存竞争问题,不能并行运行。

3.18

此题简化了一些,不用考虑car的环,使用3.17的存储做法就OK了。

3.19

这个需要个小trick,要空间常量,意味着不能存储已遍历的pair指针,那我就用时间换空间吧,使用两个指针,A指针一次步进1个pair,B指针一次步进2个pair,如果存在环,B指针会再次追上A指针,每次步进判断指针是否相等就可以知道是否带环了:)

 

posted @ 2012-02-17 16:19  effulgent  阅读(1089)  评论(0)    收藏  举报
刷新页面返回顶部
博客园  ©  2004-2025
浙公网安备 33010602011771号 浙ICP备2021040463号-3