Условные переходы

Описанные в предыдущем разделе инструкции переходов - это только часть того, что вам потребуется для написания полезных программ. В действительности необходима возможность писать такие программы, которые могут принимать решения. Именно это можно делать с помощью операций условных переходов.

Инструкция условного перехода может осуществлять или нет переход на целевую (указанную в ней) метку, в зависимости от состояния регистра флагов. Рассмотрим следующий пример:

mov ah,1 ;функция DOS ввода с клавиатуры

int 21h ; получить следующую нажатую клавишу

cmp al,'A' ; была нажата буква "A"?

je AWasTyped ; да, обработать ее

mov [TampByte],al ; нет, сохранить символ

.

.

.

AWasTyped:

push ax ; сохранить символ в стеке

Сначала в данной программе с помощью функции операционной системы DOS воспринимается нажатая клавиша. Затем для сравнения введенного символа с символом A используется инструкция CMP. Эта инструкция аналогична инструкции SUB, только ее выполнение ни на что не влияет, поскольку назначение данной инструкции состоит в том, чтобы можно было сравнить два операнда, установив флаги так же, как это делается в инструкции SUB. Поэтому в предыдущем примере флаг нуля устанавливается в значение 1 только в том случае, если регистр AL содержит символ A.

Теперь мы подошли к основному моменту. Инструкция JE представляет инструкцию условного перехода, которая осуществляет передачу управления только в том случае, если флаг нуля равен 1. В противном случае выполняется инструкция, непосредственно следующая за инструкцией JE (в данном случае - инструкция MOV). Флаг нуля в данном примере будет установлен только в случае нажатия клавиши A; и только в этом случае процессор 8086 перейдет к выполнению инструкции с меткой AWasTyped, то есть инструкции PUSH.

Набор инструкций процессора 8086 предусматривает большое разнообразие инструкций условных переходов, что позволяет вам осуществлять переход почти по любому флагу или их комбинации. Можно осуществлять условный переход по состоянию нуля, переноса, по знаку, четности или флагу переполнения и по комбинации флагов, показывающих результаты операций чисел со знаками.

Перечень инструкций условных переходов приводится в таблице 1.

Таблица 1 - Инструкции условных переходов

Название Значение Проверяемые флаги
JB/JNAE Перейти, если меньше / перейти, если не больше или равно CF = 1
JAE/JNB Перейти, если больше или равно / перейти, если не меньше CF = 0
JBE/JNA Перейти, если меньше или равно / перейти, если не больше CF = 1 или ZF = 1
JA/JNBE Перейти, если больше / перейти, если не меньше или равно CF = 0 и ZF = 0
JE/JZ Перейти, если равно ZF = 1
JNE/JNZ Перейти, если не равно ZF = 0
JL/JNGE Перейти, если меньше чем / перейти, если не больше чем или равно SF = OF
JGE/JNL Перейти, если больше чем или равно /перейти, если не меньше чем SF = OF  
JLE/JNLE Перейти, если меньше чем или равно / перейти, если не больше, чем ZF = 1 или SF = OF  
JG/JNLE Перейти, если больше чем / перейти, если не меньше чем или равно ZF = 0 или SF = OF
JP/JPE Перейти по четности PF = 1
JNP/JPO Перейти по нечетности PF = 0
JS Перейти по знаку SF = 1
JNS Перейти, если знак не установлен SF = 0
JC Перейти при наличии переноса CF = 1
JNC Перейти при отсутствии переноса CF = 0
JO Перейти по переполнению OF = 1
JNO Перейти при отсутствии переполнения OF = 0

CF - флаг переноса, SF - флаг знака, OF - флаг переполнения, ZF - флаг нуля, PF - флаг четности.

Не смотря на свою гибкость, инструкции условного перехода имеют также серьезные ограничения, поскольку переходы в них всегда короткие. Другими словами, целевая метка, указанная в инструкции условного перехода, должна отстоять от инструкции перехода не более, чем на 128 байт. Например, Турбо Ассемблер не может ассемблировать:

JumpTarget:

.

.

.

DB 1000 DUP (?)

.

.

.

dec ax

jnz JumpTarget

так как метка JumpTarget отстоит от инструкции JNZ более чем на 1000 байт. В данном случае нужно сделать следующее:

JumpTarget:

.

.

.

DB 1000 DUP (?)

.

.

.

dec ax

jnz SkipJump

jmp JumpTarget

SkipJump:

где условный переход применяется для того, чтобы определить, нужно ли выполнить длинный безусловные переход.

Командам условного перехода может предшествовать любая команда, изменяющая состояние флагов.

Пример

Рассмотрим простейший пример с использованием ввода, вывода и различных видов переходов.

Необходимо ввести с клавиатуры значение двух переменных a и x. Если а<x, то сложить их значения, а иначе из х отнять а.

model small

.386

.stack 100h

.data

a db ?

x db ?

per db 10,13,'$'

mesa db 10,13,'Input a: $'

mesx db 10,13,'Input x: $',10,13

.code

start:

mov ax, @data

mov ds, ax

lea dx, mesa

mov ah,9 ;Приглашение на ввод а

int 21h

mov ah,1 ;Считывание нажатого символа

int 21h

mov a,al ;Запись считанного символа в а

lea dx, mesx

mov ah,9 ;Приглашение на ввод х

int 21h

mov ah,1 ;Считывание нажатого символа

int 21h

mov x,al ;Запись считанного символа в х

lea dx, per

mov ah,9 ;Перевод строки

int 21h

mov al,x

cmp a,al

jl Lower ;Если а<х,то перейти на метку Lower. Иначе на метку Higher

Higher:

mov al,a

sub al,x

add al,30h ;Коррекция по вычитанию

jmp short l1

lower:

mov al,x ;В регистр al записываем результат сложения а и х

add al,a

sub al,30h ;Корекция по сложению

l1:

mov dl,al

mov ah,2 ;Вывод содержимого dl на экран

int 21h

mov ah,0 ;Ожидание нажатия клавиши

int 16h

mov ah,4ch

int 21h

end start

Остановимся подробнее на строках, в которых происходит коррекция по сложению и вычитанию. Так как Ассемблер не способен обрабатывать просто десятичные числа, то необходимо придумывать алгоритмы обработки самостоятельно. Удобнее всего приводить их к нераспакованному десятичному виду.

При чтении с клавиатуры происходит чтение именно символа, и в регистр записывается его код. Например у чисел 1 и 5 коды соответственно 31h и 35h. Чтобы привести к нераспакованному десятичному виду необходимо привести их к виду 01h и 05h. Эти коды имеют символы отличные от 1 и 5, но над ними гораздо удобнее выполнять арифметические операции. Существуют специальные команды коррекции, но о них вы узнаете позже.

Наши рекомендации