这个也是学校的课设,处理的是 0--9 的阶乘。
为什么只处理0--9的阶乘?哈哈,因为在16进制表示下9!的阶乘能被DX:AX放下啊,而且只是一位数(单个字符输入)的阶乘 ♪(^∇^*) ,多了就得考虑循环调用啥的了,有些麻烦。
简言之,往简单做,毕竟自己水平在那,不敢好高骛远,免得自找麻烦哈哈。~ o(* ̄▽ ̄*)o
设计思路:
总体设计思路和32位无符号数类似,结果存放在内存中;最主要的特点就是递归调用程序,即程序调用自己程序,最后通过比较一个界值来实现跳转。本设计的是0-9内的阶乘算法,有一个设计技巧:已知8!=40320(9D80H),9!=362880(58980H),由于进行乘法的16位运算MUL时,默认使用AX寄存器的值,乘积的结果放在DX:AX中。通过观察9!的16进制值知,在DX:AX中能放下。使用BX作为乘数源,从1往上递增。即当输入数据为9时,BX从1直到9完成递归调用。因为到8!时,AX中为9D80H,能够存放下,此时BX为9,运算后恰好能将值送到DX:AX中。如果从9递减到1的话,当BX为3时,运算结果(9X8X7X6X5X4X3=181440)对应16进制为2C4C0H,此时单用AX进行运算就会造成失真,如果要运算,需要先算DX的,再算AX的,没有必要。
输入:输入为0-9的字符。
运算(FAC):内部使用16进制进行运算。
调整(ADJUST):由于输出为10进制的运算后的数据,所以需要进行调整。内部实现是除10取余数,然后再入栈;除到最后时出栈,送到内存中去(只是为了内存首地址存储的是最高位,且按顺序存储)。
输出(OUTPUT):从内存中取出数据,调用21H的02H功能号,进行输出。
代码:
//fac.asm DATA SEGMENT BUF1 DB 0AH,0DH,'ENTER NUMBER(0-9):$' BUF2 DB 0AH,0DH,'RESULT IS :$' RESULT DB 6 DUP (?) DATA ENDS CODE SEGMENT MAIN PROC NEAR ASSUME DS:DATA,CS:CODE START: MOV AX,DATA MOV DS,AX LEA DX,BUF1 MOV AH,09H ;输出提示信息 INT 21H ;INPUT(0-9) MOV AH,01H ;输入数字n INT 21H SUB AL,30H ;ASCII转10进制 MOV BL,AL ;BL->INPUT_NUMBER MOV BH,00H MOV AX,1 XOR DX,DX MOV DI,BX MOV BX,1 ;FACT CALL FAC LEA SI,RESULT ;ADJUST CALL ADJUST INC SI MOV AL,0FFH MOV [SI],AL LEA DX,BUF2 MOV AH,09H INT 21H ;OUTPUT MOV SI,0 CALL OUTPUT MOV AH,4CH INT 21H MAIN ENDP FAC PROC NEAR CMP BX,DI JG END_FAC MUL BX INC BX CALL FAC END_FAC: RET FAC ENDP ADJUST PROC NEAR MOV CX,0 MOV BX,10 TURN_DEC: DIV BX ADD DL,30H PUSH DX INC CX CMP AX,9 MOV DX,0 JA TURN_DEC ADD AL,30H MOV [SI],AL IN_MEMORY: POP AX INC SI MOV [SI],AL LOOP IN_MEMORY RET ADJUST ENDP OUTPUT PROC NEAR LEA DI,RESULT START_OUT: MOV BL,[DI] CMP BL,0FFH JE END_OUT CMP BL,30H JE OUT_ADJUST RET_ADJUST: MOV DL,[DI] MOV AH,02H INT 21H INC DI MOV SI,1 JMP START_OUT OUT_ADJUST: CMP SI,0 JNE RET_ADJUST INC DI MOV SI,1 JMP START_OUT END_OUT: RET OUTPUT ENDP CODE ENDS END START
推荐使用notepad++或者vscode,拿来写和查看代码,感觉很棒!!!
截图:
说明:这个输出是10进制了,因为,代码中进行了除10取余的转换操作啦![]~( ̄▽ ̄)~*
嗯,就到这了,没啦!!!