Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
Программирование для начинающих 2011-09-02.pdf
Скачиваний:
45
Добавлен:
09.06.2015
Размер:
576.39 Кб
Скачать

9. Циклы while и repeat

9.1.Циклы while и repeat

Вотличие от цикла for циклы while (оператор цикла с предусловием) и repeat (оператор цикла с постусловием) позволяют повторять выполнение тех или иных операторов не фиксированное количество раз, а только пока выполняется некоторое условие. Структура записи этих операторов выглядит следующим образом:

while <условие> do begin

<Операторы> end;

<Условие> – любое логическое выражение. <Операторы> – любые операторы.

Перед каждым выполнением тела цикла анализируется значение выражения <условие>. Если оно истинно (true), выполняется тело цикла. Затем снова проверяется условие и т.д. Если значение условия ложно (false), то работа цикла завершается. Если результат условие окажется ложным при первой проверке, то тело цикла не выполнится ни разу.

Цикл repeat работает похожим образом, однако в нем условие проверяется в конце цикла (после выполнения каждого шага). Структура записи оператора выглядит следующим образом:

repeat

<Операторы> until <Условие>;

Один раз тело цикла будет выполнено в любом случае. Затем будет проверено условие и, если оно истинно, то выполнение цикла повторится. Повторение продолжается пока не выполнится условие, стоящее после слова until («пока не»). Таким образом, если в цикле while мы задаем условие для продолжения повторений, то в случае repeat‘а ставится условие на прекращение повторений.

Рис. 9.1. Блок-схемы работы циклов while (а) и repeat (б).

На рис. 9.1 работа обоих циклов показано с помощью блок-схем.

Пример 1. Имитация работы цикла for.

Циклы с условиями более универсальны, чем for. С их помощью можно легко сделать все то же самое только об изменении значения переменной счетчика придется позаботиться самостоятельно. В качестве примера напечатаем числа от 1 до 10.

i := 1;

i := 1;

while i <= 10 do

repeat

begin

writeln(i);

writeln(i);

i := i + 1;

i := i + 1;

until i > 10;

end;

 

Заметим, что в отличие от цикла for, где увеличение счетчика всегда происходит на единицу, здесь мы сами управляем счетчиком и можем организовать увеличение на любую требуемую величину. Так в следующем примере используется увеличение счетчика на 2.

Пример 2. Напечатайте все нечетные числа от 3 до 25.

i := 3;

while i <= 25 do begin

writeln(i); i := i + 2;

end;

В большинстве случаев циклы while и repeat взаимозаменяемы. Цикл repeat предпочтительнее, если известно, что тело цикла необходимо выполнить хотя бы один раз.

9.2. Зацикливание

Если логическое выражение в цикле while будет всегда истинным, то работа такого цикла не завершится никогда. Такая ситуация называется зацикливанием.

Простейший способ создать такую ситуацию:

while true do begin

writeln('У попа была собака, он ее любил.'); writeln('Она съела кусок мяса - он ее убил.'); writeln('Вырыл ямку, закопал и на камне написал:');

end;

Аналогично с repeat‘ом, только условие будет условием выхода из цикла и соответственно должно быть равно false.

Чаще, однако, такие ситуации возникают по ошибке. Для примера рассмотрим печать чисел от 1 до 10:

i := 1; repeat

writeln(i);

//i := i + 1; - Представьте, что вы забыли написать эту строку until i > 10;

Если вы забудете строку с увеличением счетчика, то i никогда не станет больше 10. Это настолько распространенная ошибка, что рекомендуется первым делом писать увеличение счетчика, а только потом возвращаться назад и писать все остальные операторы в теле цикла.

9.3. Цикл, управляемый меткой

Циклом, управляемым меткой, называется такой цикл, в теле которого на каждом шаге происходит запрос данных у пользователя, а сигналом к выходу из цикла служит ввод пользователем так называемой «метки выхода».

Для примера создадим программу, которая запрашивает у пользователя числа и подсчитывает их сумму. Количество чисел заранее не оговаривается, меткой выхода служит ввод числа 0.

s := 0; repeat

readln(x); s := s + x; until x = 0; writeln(s);

В данном случае выгоднее использовать repeat, а не while, так как хотя бы один запрос числа придется сделать. В случае while‘а этот первый запрос пришлось бы делать до цикла:

s := 0;

readln(x); {Запрос первого числа} while x <> 0 do

begin

s := s + x; readln(x);

end; writeln(s);

9.4. Вычисление номера шага

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

Пример: Коммерсант, имея стартовый капитал k рублей, занялся торговлей, которая ежемесячно увеличивает капитал на p процентов. Через сколько лет он накопит сумму s, достаточную для покупки собственного магазина?

Решение:

x := k;

{Начальная сумма равна k}

n := 0;

{Обнуляем счетчик шагов}

while x < s do {Пока сумма не достигнет s} begin

x := x * (1 + p / 100); {увеличиваем ее на p процентов} n := n + 1; {Увеличиваем счетчик шагов на 1}

end;

writeln(n div 12, ' years and ', n mod 12, ' months');

9.5. Вычисления с заданной точностью

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

Например, синус можно разложить в так называемый ряд Тейлора:

sin(x) = x x3 + x5 x7 +...

3! 5! 7!

Чем большее количество членов ряда будет просуммировано, тем точнее будет вычислен синус. Пусть требуется вычислить до 5-го знака после запятой. То есть приемлемая погрешность ε =10−5 . Для этого достаточно суммировать члены ряда до тех пор, пока очередной член ряда не окажется меньше 10−5 .

eps := 1e-5; readln(x); p := x;