|
本文成篇的起因于社长的一条记录 链接
# m# D, U9 {' O0 p8 C3 X- q" N
1 y# ~- A Z6 c, g+ E递归是计算机科学中一种解决问题的重要方法。所谓递归,通俗地说就是自己调用自己,它在很多语言和环境下都可以实现。以社长的工作场景为例,根据爱坛上的信息,大概是两个关键词:Unix和汇编。本人条件有限,只能用Linux模拟,汇编也只限于x86汇编。下面就是用汇编语言编写的,用递归方法计算阶乘的示范程序。
9 ~; M+ d) Q- X. F- LEN_OF_OUTPUT equ 10
% U" R7 j/ i7 V3 o- ^- n - $ g1 m- g! ^& n0 }. z n
- section .data ;, C8 U5 Y! D7 P$ o
- userMsg db 'Please enter a single-digit number: ' ;3 k2 W# ^$ z0 n/ N4 ^, [; l
- lenUserMsg equ $-userMsg ;- L7 W5 V4 D$ a: M
- disp1Msg db 'The factorial of '
5 p0 B0 i) ^9 k! R9 d - lenDisp1Msg equ $-disp1Msg
7 B5 q) Y. c0 i8 i1 q5 H$ w0 T - disp2Msg db ' is '
9 B6 U, l, b) \0 t( D$ R! m+ o - lenDisp2Msg equ $-disp2Msg
/ S# _( k- ~- m" g! V3 u, P - errMsg db 'Wrong input. Only one-digit number can be accepted.',0x0a
5 f, A& h" ^" ?3 y6 m% Q' C - lenErrMsg equ $-errMsg
7 w" C, _- W& {' d - ; F) G; F5 q- v/ D6 y5 ~% z
- section .bss ;
I$ e" T) A2 ]1 w - num resb 1+ N3 o! C; a0 R: d
- result resd 1
" U( T0 t! `, x t6 U - buffer resb LEN_OF_OUTPUT+1
1 g4 ~" t* W+ M% S' a% w" c -
% R+ U) Z0 z/ E8 K: Z - section .text ;
" p3 b" n6 c1 e$ c- d% d! ?% |* f - global _start
4 _) H1 D! J+ ~$ Y' Y, k -
% I" T0 g/ e; }/ M - _start: ;$ A" x, t) q3 V: |
- mov eax, 4
) m0 B( v5 S" B4 [0 b8 z* o. ~7 e( V8 r - mov ebx, 1
4 g3 n# F. n2 C$ u, v$ G9 ~ - mov ecx, userMsg
3 B# d/ M# a; l2 j3 F - mov edx, lenUserMsg8 D$ |; j% ?" r
- int 80h
3 b' R% h5 S X
f( T5 {$ @/ Q: V* M- ; read user's input
8 c( ~) d7 \+ M/ {- M% I - mov eax, 32 {7 E1 L# q" s; y9 k
- mov ebx, 0
7 g1 D1 A# Q0 h - mov ecx, num 9 ` Y9 R9 z4 K8 x5 |9 R
- mov edx, 1 ; length of input buffer7 R) J$ }& j) D
- int 80h
$ G- \* D. m! q" y, z! ?7 u# h -
% H9 }) d/ x% K+ x - mov eax, [num], V# I* C5 h+ q0 O* L
- and eax, 0xFF
: a. `: o" { c# M/ b- y - sub eax, 0x30 F4 `9 k: a# P
- jl ERROR
! W: {' l( n8 Y" f0 u$ ^ - cmp eax, 10- ?- ]& i8 B" |( L4 @+ g
- jl DO_CALC
# s4 o. d& F. _, s7 y - ERROR:% J% |& z) N0 s* p' p
- mov eax, 49 j! e" ?- R: X4 ]5 T
- mov ebx, 1
2 P1 u6 A+ \; R6 U" S9 p- m) \ - mov ecx, errMsg
2 M; W; Z3 o7 [/ k0 V! k; i$ q - mov edx, lenErrMsg
( \) ^5 V8 t/ M; I% p: r5 f - int 80h 9 b8 u3 x) t8 n y4 r
- jmp EXIT- ]! n" @/ a& R: D
- % p) \, E' T" l7 A$ {* o* r
- - e* n; U8 X% D8 I: M! l
- DO_CALC:
2 x: W& r( Y- u' @ - call factorial1 {8 C2 o4 g- w8 K
- ; The parameter is in eax. After calling, the result will be stored in eax.
* t1 y- J$ V2 e5 r' j# [ - mov [result], eax ; Save the result
: z2 a% ^. Z) E1 L+ p* ?2 M# m4 w
9 R, g; t5 b" m0 E8 t- ; print result
1 S: E% p% S0 l; }5 e - mov eax, 40 X3 u. K* I" T) B% E# c
- mov ebx, 17 t0 r/ g# W; S$ K
- mov ecx, disp1Msg# M# \) l. p+ |* Y+ n" I
- mov edx, lenDisp1Msg
; n4 z2 }4 d2 ?6 i - int 80h
2 G' m8 m$ v; M: B1 c/ N5 V - mov eax, 4
' K$ ^/ b; U, O0 Y' O - mov ebx, 1
' s" g5 e8 {+ H% O! ~4 O! I1 x' [ - mov ecx, num
5 \7 q& [3 |3 ` - mov edx, 1
: Q4 |" o4 L7 C- J" v! o0 ~ - int 80h
+ G1 t8 J" l' O - mov eax, 47 {; ?, b0 i9 Q* d/ G6 h. a. Z% X8 {
- mov ebx, 1
C2 C( w9 f* u2 L- J, g9 V. T - mov ecx, disp2Msg4 q7 B8 K3 t) y {7 i) f6 S, r
- mov edx, lenDisp2Msg
: k$ Q$ c7 K; E' s! [4 { - int 80h & D0 }: l9 K& v% K
2 F; B6 T" L! H* Q& h* m- ; output number( P" Y) y, T3 ~) T! r" k
- mov eax, [result] ; VALUE THAT I WANT TO CONVERT AND DISPLAY
+ O% ^0 S/ n7 r1 J/ }5 E8 Q, T - mov byte [buffer+LEN_OF_OUTPUT],0x0a
+ h* L. V% v/ m6 O0 ]- s; \0 g - lea esi,[buffer+LEN_OF_OUTPUT]; j5 u" L2 U; R# O8 e3 F
- mov ebx,10
4 U; B1 I6 V1 C( `8 L& U - ASC_LOOP:
- n9 b% g- z* t. a - mov edx,0 ; clear dx prior to dividing edx:eax by bx
* ?! L: K S7 h0 K1 D - div ebx ;DIV EAX/107 n4 M# | A1 F0 }7 s8 m. e! \: k$ J
- add edx,0x30 ;ADD 48 TO REMAINDER TO GET ASCII CHARACTER OF NUMBER " l& l7 F, ^* @0 P0 }
- dec esi ; store characters in reverse order ~* u$ j- ^; d( g# ?! Z
- mov [esi],dl1 Z: f K' Z' u! v" a/ ?
- cmp eax,0 $ }, Q; p; `( [* h+ T7 ~5 f5 t- [
- jz END_ASC ;IF AX=0, END THE PROCEDURE OF NUMBER TO ASCII: Z# ]2 ~1 ?1 c5 n6 l v: \% o
- jmp ASC_LOOP ;ELSE REPEAT
}. G ~8 f. f8 W B - END_ASC:
8 _% t% o- q1 { - mov eax, 4- X5 |9 K. u4 F7 s* c* S
- mov ebx, 1
2 [; J+ ]+ M1 e; n$ G: n; u9 M - mov ecx, esi
2 n5 k# W1 G1 y- ^3 x. D - lea edx,[buffer+LEN_OF_OUTPUT+1]
9 m& x& I4 x0 c3 m) W - sub edx, ecx1 v9 f1 H) g4 F$ t l. _
- int 80h 4 v* o8 v/ j- b; X% n
-
! F3 o# i! a! | - ; Exit code
1 b3 N- V! v# l/ e& K5 A - EXIT:
" n3 `# Q6 ?$ [! b4 Q- W: { - mov eax, 1
! ^4 U, F+ }6 t5 r1 e - mov ebx, 02 k4 @( B) A* C% S, `! m8 L
- int 80h# U% {) x% O# i7 x1 T
- N6 ]8 F; r6 r, B3 E& P% R& O# a- : H9 U) m9 s- V& k
/ p; M' j {+ D9 s# A- ; Factorial function using recursion$ B0 L5 W+ w( U/ P; S
- factorial:
2 [1 E+ e/ v$ P: X3 o% `1 f - ; Check if eax is 0 (base condition)
! p! ?6 p8 V# q$ X2 `& P# v! K) R, P7 R - cmp eax, 0! ]1 h) S( l+ N* `- \1 y, D
- jz end_recursion" o1 \7 I7 f! ^8 i% S4 t) {
- 3 j4 e1 @- x: o/ b1 U8 c
- ; Save the current value of eax u+ W# C5 m" p1 G+ }; r5 o
- push eax
3 N8 i8 Z9 z" u* P - & X3 `3 o9 S7 c8 V: P; S7 q
- ; Decrement eax and call factorial function recursively
+ d3 t5 R( M8 C - dec eax3 \- {0 B4 ~( ]/ z2 a$ K( j7 u4 L
- call factorial
9 p) s9 \1 y6 J
0 W) S& ], I. G( h; T1 |: }& p- ; Multiply the result returned in eax with the saved value of eax
( Q4 i: E0 a3 [# o - pop ebx+ J5 w% M8 h6 x. A- N3 L7 k: }
- imul eax, ebx& x7 K% E; L t- c- X) A
+ R! Z; ?3 P+ s' l8 |( p& A& Q- ret( p5 K( u7 n$ ^1 G+ k. I
- ( O: C3 ?' j7 n$ P& z& g$ r& P. s
- end_recursion:
* T! q! y5 t0 C- `) g" p: U6 _ - ; Return 1 when eax is 0 (base condition)
! R# S7 R% m! O2 Y, e - mov eax, 1# d/ }2 z5 S1 m6 i) @, j
- ret
复制代码
7 U/ a1 f% q! Y3 d" u6 l程序在nasm编译器下通过并成功运行。相应的命令行如下,有兴趣的童鞋可以自行验证。: I8 f5 s" O+ o7 [8 L) m B
- nasm -f elf factorial.asm' z% R0 `* O; J; A( }& r9 }
- ld -m elf_i386 -s -o factorial factorial.o$ A' u: ?' z/ f
- ./factorial" y/ C0 q4 M/ h0 d3 M* \
复制代码 , s2 R( e- k2 v& X; u7 }* o+ S
由于汇编不擅长处理I/O,所以程序限制输入仅限于一位数字(0-9),以免过于喧宾夺主。其实程序中真正紧扣主题的就最后一小段:. C; V/ ?2 w0 h1 c
- ; Factorial function using recursion; |! O1 L7 u0 `3 Z" }
- factorial:
3 O+ K' Y2 I% A% Z7 r( M" ? - ; Check if eax is 0 (base condition)) x! V1 G& r" f/ J
- cmp eax, 0
& I9 E- ~" G2 i4 G, h" F - jz end_recursion8 J; a1 b8 {! C; I2 r9 O
- # x9 |9 D& s$ O4 @; c
- ; Save the current value of eax. ~$ P. R# j4 L
- push eax) N; c& [. A8 F: Y
- + L( _# K: z# s6 d3 T6 U8 M
- ; Decrement eax and call factorial function recursively
3 K' k6 b* N5 S2 `0 B1 `2 H - dec eax
0 q1 I7 X8 w( F! O2 j# z - call factorial d- u0 R* I3 Y+ Q+ i
- ; F3 H3 y6 t1 _8 M
- ; Multiply the result returned in eax with the saved value of eax
3 r8 M3 P* g! t! @" J" @/ Z- W - pop ebx
9 O2 X% R0 w, A) T) o) [ - imul eax, ebx9 Y7 b/ t4 l* \9 x" X' n/ Q
5 l+ O! h2 e9 P) P6 z3 z% u* X- ret& M/ k9 i$ q+ l/ K/ L0 \# S2 X
7 t6 \1 U- U- L9 y% F- end_recursion:
. C# W# E3 P5 l+ _ - ; Return 1 when eax is 0 (base condition)% F5 p C/ L3 e0 D
- mov eax, 1
: o( V1 j) q& q% \) d$ i& Z - ret
复制代码 可以清晰看出函数在其函数体中调用了它本身。" A5 p2 H. n( i
8 l2 o. S/ i2 M4 J
以上证明了汇编语言不是实现递归的障碍。推而广之,所有中低级语言都可以轻松实现递归。某些高级语言反而不能做递归是人为强制规定的,换而言之是原作者权衡利弊的结果。
( Y( S9 ~' Q" y& u; p; V3 d; h4 O
. t/ V: I- w% \+ a6 b9 @: `+ T世界上没有免费的午餐,递归在编写程序的时候简洁优雅,但运行过程中要付出极大的代价,甚至有可能是灾难。由于是函数反复调用自己,那么所有的中间结果都要暂时保留在栈(stack)上,直到最后一层函数调用完毕才能逐步清除。这既低效,又不安全。栈式计算是一个古老的概念,它与现代的多寄存器和流水线等都格格不入,难以优化提速。递归对栈空间的占用更可怕,占用量与调用函数次数线性相关(套用复杂度的概念就是O(n)),递归次数多了之后,很容易超出栈容量,从而导致栈溢出。具体到社长的工作中可能还有很多实时任务,一大堆高优先级的中断响应程序可不会等着递归函数运行完了再启动,这会让栈空间雪上加霜。想一想某个核电控制程序突然栈溢出了,那场面,哈哈哈。+ Z6 p5 x" P# E/ _# s% H" X3 E
. q0 `8 y* |6 b% B9 T" C; ?* c) {这可以解释,为什么条件上有可能,但社长很长时间里都没有接触到递归编程。无论用C还是用汇编,都强调的是速度与可靠性。社长实战出发,不需要知道回字的4种写法。至于Python,整个编程思想都变了,Python假定计算机的计算速度远远超过了需求,编程者无需为性能而操心。 Python语言的设计初衷,也不是为关键任务而生,而是想让更多的门外汉用起计算机来。所以在Python程序中容易见到递归。
& D7 t l" s: ?9 U
: N2 }; s# P% r& R6 Q有很多时候,递归用作概念性算法表达。真到了用代码实现的时候,再换用其它方法。比如说将递归转化成迭代(iteration)。$ o& v- A- a& S9 f/ ~
7 U0 q3 B5 M% o9 ?4 D0 Y另外,汇编语言作为一种低级编程语言,在不考虑时间成本的前提下,没有什么是它做不了的。本期课后作业:如何用汇编语言实现继承(Inheritance)和多态(Polymorphism)?请写出示范程序。我们知道继承和多态是面向对象编程(Object-oriented programming)的两个基本点,有了它们就可以实现汇编语言的面向对象编程。
) B6 f* x& W; B5 j( }; \% N) Z `6 N# o7 T" X0 U3 m
|
评分
-
查看全部评分
|