明经CAD社区

 找回密码
 注册

QQ登录

只需一步,快速开始

搜索
查看: 4329|回复: 8

[讨论] 递归 — 从斐波那契数列的Lisp求法开始

[复制链接]
发表于 2013-6-28 03:46:55 | 显示全部楼层 |阅读模式
本帖最后由 yxp 于 2013-6-28 15:14 编辑

收集和总结一下递归的用法,不足之处请大家指正。
从下面例子可以看出递归的算法如同数学归纳法,在函数内部会一直倒推至n=0(1)的值,如果不能收敛(缺少递归结束条件),就会出现堆栈溢出。因此对于短表尚可,长表运行效率较低。


斐波那契数列也叫黄金分割数列,从第三项开始值为前两项之和 '(1 1 2 3 5 8 13 21 34 55)
递归1,求斐氏数列第n项值
示例:  (Fib 12)  返回 144
(defun Fib(n)(if (and (> n 0)(< n 3)) 1 (+ (Fib (- n 1))(Fib (- n 2)))))


递归2,删除L表中的重复元素
示例: (delSame '(1 3 2 4 2 3 5))   返回 (1 3 2 4 5)
(defun delSame (L) (if L (cons (car L) (delSame (vl-remove (car L) L)))))


递归3,删除L表的n个元素
示例: (delFnth 3 '(1 2 3 4 5 6))   返回 (4 5 6)
(defun delFnth (n L)(if (= n 0) L (delFnth (1- n) (cdr L))))


递归4,返回L表的前n个元素
示例: (getFnth 3 '(1 2 3 4 5 6))   返回 (1 2 3)
(defun getFnth (n L)(if (= n 0) nil (cons (car L) (getFnth (1- n) (cdr L)))))


递归5,删除L表的n个元素
示例:(Delnth 2 '(1 2 3 4 5 6)  返回 (1 2 4 5 6)
(defun Delnth (n L)(if (= n 0)(cdr L)(cons (car L) (Delnth (1- n) (cdr L)))))


递归6,更新L表的第n个元素va
示例: (Renth 0 2  '(1 2 3 4 5 6))  返回 (1 2 0 4 5 6)
(defun Renth (va n L)(if (= n 0)(cons va (cdr L))(cons (car L) (Renth va (1- n) (cdr L)))))



递归7,插入元素va到L表的第n位
示例: (Insnth 0 2 '(1 2 3 4 5 6))  返回 (1 2 0 3 4 5 6)
(defun Insnth (va n L)(if (= n 0)(cons va L) (cons (car L) (Insnth va (1- n) (cdr L)))))



其他组合应用,例如:返回L表的m到n位,不分m,n的大小
示例: (getList 2 4 '(1 2 3 4 5 6))  返回 (3 4 5)
(defun getList(m n L)(getFnth (+ (abs (- m n)) 1) (delFnth (min m n) L)))

想想数学归纳法,递归的结构就会比较清晰,对表其他递归操作可以此类推。
对表的操作基本上都以 n=0,或者空表作为递归结束条件,以 1- 或 cdr 作为循环条件,循环语句的结构极其简单,且与 n=0 时的结构相同;
从n开始调用自身,当满足 n=0 条件时,便不再循环调用自身;
大部分递归结构都可以用循环算法代替。

以上递归基本是调用组合 cons、car、cdr 函数,所以对于短表操作效率还是很高的。
据说,nth 函数本身就是 car 和 cdr 内部函数的递归表达式。相当于:
(defun Xnth (n L)(if (= n 0)(car L)(Xnth (1- n) (cdr L))))
测试: (Xnth 3 '(1 2 3 4 5 6)) 返回 4
简单吧。再比如
(defun Xlength (L)(if L (1+ (Xlength (cdr L))) 0))
结束条件为空表,返回0,否则返回 1+
测试: (Xlength '(1 2 3 4 5 6))  返回 6

如果你看懂了,可以试着写一下:
1 求自然数L表前n个元素之和
  如 (nSum 3 '(7 24 12 11 2 5)) 返回 43
2 在L表中间隔插入va值
  如 (InsV '(1 2 3 4 5) "a") 返回 (1 "a" 2 "a" 3 "a" 4 "a" 5)


评分

参与人数 4明经币 +7 金钱 +34 收起 理由
tigcat + 10 很给力!
xyp1964 + 3 鼓励
twsyzx + 1 很给力!数学果然是万科的妈妈
Gu_xl + 3 + 24 赞一个!

查看全部评分

"觉得好,就打赏"
还没有人打赏,支持一下

本帖被以下淘专辑推荐:

相关帖子

发表于 2013-6-28 09:10:16 | 显示全部楼层
递归是个很值得研究的问题,理论上使用递归很简单,但往往使用不好,就造成堆栈溢出。
现在关于递归还有一个“尾递归”的概念,就是为了解决堆栈问题,道理看似很清楚,想用却并不容易,楼主部分也顺便研究一下。
发表于 2013-12-25 00:43:29 | 显示全部楼层
这么好的文章,看到的晚了

浮起让更多的人受益
发表于 2014-3-7 14:06:07 | 显示全部楼层
确实是好东西,先收藏,打起了精神再学习.
发表于 2014-12-18 13:56:29 | 显示全部楼层
我用递归2的方法(delSame )处理点表时怎么进入死循环,跳不出来。
发表于 2014-12-18 20:09:43 | 显示全部楼层
好文章,学习。
发表于 2015-3-8 15:46:45 | 显示全部楼层
我是做题的:

; 求自然数L表前n个元素之和
  ;如 (nSum 3 '(7 24 12 11 2 5)) 返回 43
(defun nsum(n l) (if (= n 0) 0        (+ (car l) (nsum (1- n) (cdr l)))))

;在L表中间隔插入va值
; 如 (InsV '(1 2 3 4 5) "a") 返回 (1 "a" 2 "a" 3 "a" 4 "a" 5)
(defun insv(l va) (if (> (length l) 2) (cons (car l) (cons va  (insv (cdr l) va))) (cons (car l) (cons va (cdr l)))))
发表于 2020-12-24 19:28:57 | 显示全部楼层
递归1
(and (> n 0)(< n 3))
改成 (and (< n 3) (> n 0))  因为一般输入都是大于3的,就不需要判断>0了
写成(< 0 n 3) 会好看点
发表于 2020-12-24 21:13:46 | 显示全部楼层
全凭测试出来的..难啊.
(defun nSum (n lst)
  (if (= n 1)
    (car lst)
    (+ (car lst) (nSum (1- n) (cdr lst)))
  )
)
(nSum 3 '(7 24 12 11 2 5))

(defun InsV (lst va)
  (if (cdr lst)
    (cons (car lst) (cons va (InsV (cdr lst) va)))
    (list (car lst))
  )
)
(InsV '(1 2 3 4 5) "a")
您需要登录后才可以回帖 登录 | 注册

本版积分规则

小黑屋|手机版|CAD论坛|CAD教程|CAD下载|联系我们|关于明经|明经通道 ( 粤ICP备05003914号 )  
©2000-2023 明经通道 版权所有 本站代码,在未取得本站及作者授权的情况下,不得用于商业用途

GMT+8, 2024-11-13 14:52 , Processed in 0.199635 second(s), 31 queries , Gzip On.

Powered by Discuz! X3.4

Copyright © 2001-2021, Tencent Cloud.

快速回复 返回顶部 返回列表