我如何判断我的函数是否已经过尾调用优化?
How can I tell if my function has been tail call optimized?
我正在阅读 On Lisp 中的第 2.8 节(尾递归)。它有一个尾递归函数的例子:
(defun our-length-tr (lst)
"tail recursive version with accumulator"
(labels ((rec (lst acc)
(if (null lst)
acc
(rec (cdr lst) (1+ acc)))))
(rec lst 0)))
它说 许多 Common Lisp 编译器执行 TCO,但您可能需要在文件顶部添加 (proclaim '(optimize speed))
。
我如何确定我的编译器支持 TCO,以及它将我的函数编译为循环版本而不是递归版本?
有几种简单的方法可以检查函数是否使用尾递归编译。
如果你能看懂汇编语言那么原始函数disassemble
(见documentation)就可以使用,例如:
* (disassemble 'our-length-tr)
; disassembly for OUR-LENGTH-TR
; Size: 89 bytes. Origin: #x10034F8434
; 34: 498B4C2460 MOV RCX, [R12+96] ; no-arg-parsing entry point
; thread.binding-stack-pointer
; 39: 48894DF8 MOV [RBP-8], RCX
; 3D: 488B4DF0 MOV RCX, [RBP-16]
; 41: 31D2 XOR EDX, EDX
; 43: EB3E JMP L2
; 45: 660F1F840000000000 NOP
; 4E: 6690 NOP
; 50: L0: 4881F917001020 CMP RCX, #x20100017 ; NIL
; 57: 7506 JNE L1
; 59: 488BE5 MOV RSP, RBP
; 5C: F8 CLC
; 5D: 5D POP RBP
; 5E: C3 RET
; 5F: L1: 8D41F9 LEA EAX, [RCX-7]
; 62: A80F TEST AL, 15
; 64: 751F JNE L3
; 66: 488B5901 MOV RBX, [RCX+1]
; 6A: 48895DE8 MOV [RBP-24], RBX
; 6E: BF02000000 MOV EDI, 2
; 73: 41BBF004B021 MOV R11D, #x21B004F0 ; GENERIC-+
; 79: 41FFD3 CALL R11
; 7C: 488B5DE8 MOV RBX, [RBP-24]
; 80: 488BCB MOV RCX, RBX
; 83: L2: EBCB JMP L0
; 85: L3: 0F0B0A BREAK 10 ; error trap
; 88: 2F BYTE #X2F ; OBJECT-NOT-LIST-ERROR
; 89: 08 BYTE #X08 ; RCX
; 8A: 0F0B10 BREAK 16 ; Invalid argument count trap
NIL
(Mac OS X 10.13.3 上的 SBCL 1.4.1)
否则你可以用一个很长的列表调用函数,看看结果是Stack Overflow错误(递归编译为递归),还是列表的长度(递归用迭代编译,尾递归)。
更好的是,您可以提供无限长度的列表,例如:
(our-length-tr '#1=(1 2 3 . #1#)))
并查看是否产生了堆栈溢出错误(通常几乎立即),或者由于迭代的无限循环而根本没有产生任何输出。
我正在阅读 On Lisp 中的第 2.8 节(尾递归)。它有一个尾递归函数的例子:
(defun our-length-tr (lst)
"tail recursive version with accumulator"
(labels ((rec (lst acc)
(if (null lst)
acc
(rec (cdr lst) (1+ acc)))))
(rec lst 0)))
它说 许多 Common Lisp 编译器执行 TCO,但您可能需要在文件顶部添加 (proclaim '(optimize speed))
。
我如何确定我的编译器支持 TCO,以及它将我的函数编译为循环版本而不是递归版本?
有几种简单的方法可以检查函数是否使用尾递归编译。
如果你能看懂汇编语言那么原始函数disassemble
(见documentation)就可以使用,例如:
* (disassemble 'our-length-tr)
; disassembly for OUR-LENGTH-TR
; Size: 89 bytes. Origin: #x10034F8434
; 34: 498B4C2460 MOV RCX, [R12+96] ; no-arg-parsing entry point
; thread.binding-stack-pointer
; 39: 48894DF8 MOV [RBP-8], RCX
; 3D: 488B4DF0 MOV RCX, [RBP-16]
; 41: 31D2 XOR EDX, EDX
; 43: EB3E JMP L2
; 45: 660F1F840000000000 NOP
; 4E: 6690 NOP
; 50: L0: 4881F917001020 CMP RCX, #x20100017 ; NIL
; 57: 7506 JNE L1
; 59: 488BE5 MOV RSP, RBP
; 5C: F8 CLC
; 5D: 5D POP RBP
; 5E: C3 RET
; 5F: L1: 8D41F9 LEA EAX, [RCX-7]
; 62: A80F TEST AL, 15
; 64: 751F JNE L3
; 66: 488B5901 MOV RBX, [RCX+1]
; 6A: 48895DE8 MOV [RBP-24], RBX
; 6E: BF02000000 MOV EDI, 2
; 73: 41BBF004B021 MOV R11D, #x21B004F0 ; GENERIC-+
; 79: 41FFD3 CALL R11
; 7C: 488B5DE8 MOV RBX, [RBP-24]
; 80: 488BCB MOV RCX, RBX
; 83: L2: EBCB JMP L0
; 85: L3: 0F0B0A BREAK 10 ; error trap
; 88: 2F BYTE #X2F ; OBJECT-NOT-LIST-ERROR
; 89: 08 BYTE #X08 ; RCX
; 8A: 0F0B10 BREAK 16 ; Invalid argument count trap
NIL
(Mac OS X 10.13.3 上的 SBCL 1.4.1)
否则你可以用一个很长的列表调用函数,看看结果是Stack Overflow错误(递归编译为递归),还是列表的长度(递归用迭代编译,尾递归)。
更好的是,您可以提供无限长度的列表,例如:
(our-length-tr '#1=(1 2 3 . #1#)))
并查看是否产生了堆栈溢出错误(通常几乎立即),或者由于迭代的无限循环而根本没有产生任何输出。