2.4. Вложенные циклы Поскольку тело любого цикла состоит из операторов или составного оператора, то ничто не помешает в теле цикла располагать другие операторы цикла. Рассмотрим работу программы: Program Prim21; Var i,j:integer; BEGIN for i:=1 to 5 do Begin writeln; for j:=20 to 23 do write('i=',i,' j=',j); end; readln; Для цикла for i:=1 to 5 do телом цикла является begin for j:=20 to 23 do write(' i= ', i, ' , j = ', j); writeln; end; Назовем этот цикл внешним. Кроме того, имеется внутренний цикл for j: = 20 to 23 do с телом write (' i = ', i , j =', j); Работа программы. Заметим, что оператор write отличается от оператора writeln тем, что он не начинает вывод с новой строки, а продолжает писать в той же строке, т.е. после второго выполнения внутреннего цикла на экране появится i= 1, j=20 i= 1, j=21. Машина продолжит выполнение внутреннего цикла, и, когда он закончится (выполнится для j = 20.21.22.23), на экране будет строка i = 1 j = 20 i =l j = 21 i = 1 j = 22 i = 1 j = 23. Внутренний цикл закончится, однако тело внешнего цикла еще не закончилось, поэтому выполняется оператор writeln, который переводит курсор на новую строку. После этого тело внешнего цикла закончится, но сам цикл отработал только для i = 1. Поэтому внешний цикл продолжит работу, присвоив i: =2 и вновь начав выполнение своего тела. Встретив внутренний цикл j:=1, на экран с новой строки выведется: i=2, j=20, затем j:=2 и к этой строке добавится i=2, j=21 и т.д., пока не закончится внутренний цикл. Таким образом, внешний цикл, изменяя i от 1 до 5, заставит каждый раз выполняться полностью внутренний цикл, и в результате работы программы на экране появится: i=l, j=20 i=1, j=21 i=1, j=22 i=1, j=23 i=2, j=20 i=2, j=21 i=2, j=22 i=2, j=23 i=3, j=20 i=3, j=21 i=3, j=22 i=3, j=23 i=4, j=20 i=4, j=21 i=4, j=22 i=4, j=23 i=5, j=20 i=5, j=21 i=5, j=22 i=5, j=23 Вкладывать циклы друг в друга можно любое количество раз, необходимо лишь помнить, что количество выполнений самого внутреннего тела цикла при этом будет расти в геометрической прогрессии. Так, например, for i:=l to 100 do for j:=l to 100 do for k:=l to 100 do writeln (i, j, k); дает столбик цифр: 111 112 113 114 - - - - - 121 122 123 124 - - - - - 211 212 213 - - - - - 100100100, что составляет 1000000 строчек. Пример: найти rez = f (х1) + f (х2 )+.. .+f (x6), где f(x) считать с точностью не менее e=0.01; х1=0.1, затем xn=xn-l+0.2, пока х не станет равным 1.1. Нам нужно вычислить rez. Поскольку это будет сумма, то вначале положим rez=0, затем вычислим f(х1) с нужной точностью и добавим к rez, и т.д., то есть: Rez : = 0 ; x:=0.l; for i:=l to 6 do Begin x:=x+0.2; end; Теперь в данную программу нужно "вставить" вычисленные f(x) с необходимой точностью. Для этого найдем сумму, составляющую f(x) для n элементов, затем - для n+l элемента, сравним их по модулю и, когда разность между ними будет меньше Е, сумму для n+l элемента ряда примем за более точную и добавим к Rez. При вычислении f(x) количество членов ряда, которые нам придется суммировать, не известны, поэтому придется применять цикл типа repeat или while. Кроме того, в элементы ряда входит 2n! Факториал подсчитывается по схеме 1! = 1, 2! = 1*2; 3! =1*2*3 и т.д., т.е. например, 8! = 12345678. Далее, при подсчете следующего значения ряда, можно величину 2n! найти по схеме: fak: = fak(n-l)n. Ряд у нас знакопеременный, т.е. меняется '+', '-' перед каждым членом ряда. Меняем знак с помощью схемы znak: = znak(-1). Если исходный знак = +l, то в ячейке znak будет храниться то +l, то -1. С учетом сказанного выше f(x) с нужной точностью вычисляется: fxpred: = 0; fxpos: =l, n: = 2; znak: = 1; fak: = 1; while Abs(fxpos - fxpred) > = 0.01 do Begin Соединив оба фрагмента программы и снабдив их описаниями, получим готовую программу: Program Prim22; Var n,fak,i,znak:longint; rez,x,fxpred,fxpos:real; BEGIN for i:= 1 to 6 do Begin while abs(fxpos-fxpred)>=0.001 do Begin writeln('rez=',rez); readln; END. |