Опубликован: 08.04.2009 | Уровень: для всех | Доступ: платный
Лекция 1:

Переменные, выражения, присваивания

1.2.10. Дан массив x[1]..x[n] целых чисел. Не используя других массивов, переставить элементы массива в обратном порядке.

Решение. Элементы x[i] и x[n+1-i] нужно поменять местами для всех i, для которых i<n+1-i, то есть

$\w{2}\w{i} < \w{n} +
\w{1}$~$\Leftrightarrow$
$\w{2}\w{i}\le \w{n}\hm\Leftrightarrow \w{i} \le \text{n div 2}$

for i := 1 to n div 2 do begin 
| ...поменять местами  x[i] и x[n+1-i]; 
end;

1.2.11. (Из книги Д. Гриса) Дан массив целых чисел x[1]..x[m+n], рассматриваемый как соединение двух его отрезков: начала x[1]..x[m] длины m и конца x[m+1]..x[m+n] длины n. Не используя дополнительных массивов, переставить начало и конец. (Число действий порядка m+n

Решение. Вариант 1. Перевернем (расположим в обратном порядке) отдельно начало и конец массива, а затем перевернем весь массив как единое целое.

Вариант 2. (А. Г. Кушниренко) Рассматривая массив записанным по кругу, видим, что требуемое действие - поворот круга. Как известно, поворот есть композиция двух осевых симметрий.

Вариант 3. Рассмотрим более общую задачу - обмен двух участков массива x[p+1]..x[q] и x[q+1]..x[r]. Предположим, что длина левого участка (назовем его A ) не больше длины правого (назовем его B ). Выделим в B начало той же длины, что и A, назовем его B_1, а остаток B_2. (Так что B=B_1+B_2, если обозначать плюсом приписывание массивов друг к другу.) Нам надо из A+B_1+B_2 получить B_1+B_2+A. Меняя местами участки A и B_1 - они имеют одинаковую длину, и сделать это легко, - получаем B_1 + A + B_2, и осталось поменять местами A и B_2. Тем самым мы свели дело к перестановке двух отрезков меньшей длины. Итак, получаем такую схему программы:

p := 0; q := m; r := m + n; 
{инвариант: осталось переставить x[p+1..q], x[q+1..r]} 
while (p <> q) and (q <> r) do begin 
| {оба участка непусты} 
| if (q - p) <= (r - q) then begin 
| | ..переставить x[p+1]..x[q] и x[q+1]..x[q+(q-p)] 
| | pnew := q; qnew := q + (q - p); 
| | p := pnew; q := qnew; 
| end else begin 
| | ..переставить x[q-(r-q)+1]..x[q] и x[q+1]..x[r] 
| | qnew := q - (r - q); rnew := q; 
| | q := qnew; r := rnew; 
| end; 
end;

Оценка времени работы: на очередном шаге оставшийся для обработки участок становится короче на длину A ; число действий при этом также пропорционально длине A.

1.2.12.Коэффициенты многочлена лежат в массиве a: array[0..n] of integer ( n - натуральное число, степень многочлена). Вычислить значение этого многочлена в точке x, то есть a[n]\,x^n+\ldots+a[1]\,x+a[0].

Решение. (Описываемый алгоритм называется схемой Горнера.)

k := 0; y := a[n]; 
{инвариант: 0 <= k <= n, 
 y= a[n]*(x в степени k)+...+a[n-1]*(x в степени k-1)+...+ 
+ a[n-k]*(x в степени 0)} 
while k<>n do begin 
| k := k + 1; 
| y := y * x + a [n-k]; 
end;

1.2.13. (Для знакомых с основами анализа; сообщил А. Г. Кушниренко) Дополнить алгоритм вычисления значения многочлена в заданной точке по схеме Горнера вычислением значения его производной в той же точке.

Решение. Добавление нового коэффициента соответствует переходу от многочлена P(x) к многочлену xP(x)+c. Его производная в точке x равна xP'(x) + P(x). (Это решение обладает забавным свойством: не надо знать заранее степень многочлена. Если требовать выполнения этого условия, да еще просить вычислять только значение производной, не упоминая о самом многочлене, получается не такая уж простая задача.)

Общее утверждение о сложности вычисления производных таково:

1.2.14. (В. Баур, Ф. Штрассен) Дана программа вычисления значения некоторого многочлена P(x_1,\ldots,x_n), содержащая только команды присваивания. Их правые части - выражения, содержащие сложение, умножение, константы, переменные x_1,\ldots,x_n и ранее встречавшиеся (в левой части) переменные. Доказать, что существует программа того же типа, вычисляющая все n производных \partial P/\partial x_1,\ldots, 
\partial P/\partial x_n, причем общее число арифметических операций не более чем в C раз превосходит число арифметических операций в исходной программе. Константа C не зависит от n.

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

1.2.15.В массивах a: array[0..k] of integer и b: array[0..l] of integer хранятся коэффициенты двух многочленов степеней k и l. Поместить в массив c: array[0..m] of integer коэффициенты их произведения. (Числа k,l,m - натуральные, m=k+l ; элемент массива с индексом i содержит коэффициент при степени i.)

Решение.

for i:=0 to m do begin 
| c[i]:=0; 
end; 
for i:=0 to k do begin 
| for j:=0 to l do begin 
| | c[i+j] := c[i+j] + a[i]*b[j]; 
| end; 
end;

1.2.16. Предложенный выше алгоритм перемножения многочленов требует порядка n^2 действий для перемножения двух многочленов степени n. Придумать более эффективный (для больших n ) алгоритм, которому достаточно порядка n^{log 4/log 3} действий.

Указание. Представим себе, что надо перемножить два многочлена степени 2k. Их можно представить в виде

A(x)\,x^k + B(x) \quad  и \quad   C(x)\,x^k + D(x).
Произведение их равно
A(x)C(x)\,x^{2k}  +  (A(x)D(x)+B(x)C(x))\,x^k  + B(x)D(x).
Естественный способ вычисления AC, AD+BC, BD требует четырех умножений многочленов степени k, однако их количество можно сократить до трех с помощью такой хитрости: вычислить AC, BD и (A+B)(C+D), а затем заметить, что AD+BC=(A+B)(C+D)-AC-BD.

1.2.17.Даны два возрастающих массива x: array[1..k] of integer и y: array[1..l] of integer. Найти количество общих элементов в этих массивах, то есть количество тех целых t, для которых t = x[i] = 
y[j] для некоторых i и j. (Число действий порядка k+l.)

Решение.

k1:=0; l1:=0; n:=0; 
{инвариант: 0<=k1<=k; 0<=l1<=l; 
 искомый ответ = n + количество общих 
 элементов в x[k1+1]...x[k] и y[l1+1]...y[l]} 
while (k1 <> k) and (l1 <> l) do begin 
| if x[k1+1] < y[l1+1] then begin 
| | k1 := k1 + 1; 
| end else if x[k1+1] > y[l1+1] then begin 
| | l1 := l1 + 1; 
| end else begin {x[k1+1] = y[l1+1]} 
| | k1 := k1 + 1; 
| | l1 := l1 + 1; 
| | n := n + 1; 
| end; 
end; 
{k1 = k или l1 = l, поэтому одно из множеств, упомянутых 
 в инварианте, пусто, а n равно искомому ответу}

Замечание. В третьей альтернативе достаточно было бы увеличивать одну из переменных k1, l1 ; вторая добавлена для симметрии.

1.2.18.Решить предыдущую задачу, если про массивы известно лишь, что x[1]\le\ldots\le x[k] и y[1]\le\ldots\le y[l] (возрастание заменено неубыванием).

Решение. Условие возрастания было использовано в третьей альтернативе выбора: сдвинув k1 и l1 на 1, мы тем самым уменьшали на 1 количество общих элементов в x[k1+1]\ldots x[k] и x[l1+1]\ldots x[l]. Теперь это придется делать сложнее.

... 
end else begin {x[k1+1] = y[l1+1]} 
| t := x [k1+1]; 
| while (k1<k) and (x[k1+1]=t) do begin 
| | k1 := k1 + 1; 
| end; 
| while (l1<l) and (x[l1+1]=t) do begin 
| | l1 := l1 + 1; 
| end; 
| n := n + 1; 
end;

Замечание. Эта программа имеет дефект: при проверке условия

$$
\w{(k1<k) and (x[k1+1]=t)}
$$
(или второго, аналогичного) при ложной первой скобке вторая окажется бессмысленной (индекс выйдет за границы массива) и возникнет ошибка. Некоторые версии паскаля, вычисляя A and B, сначала вычисляют A и при ложном A не вычисляют B. (Так ведет себя, например, система Turbo Pascal версии 5.0 - но не 3.0.) Тогда описанная ошибка не возникнет.

Но если мы не хотим полагаться на такое свойство используемой нами реализации паскаля (не предусмотренное его автором Н. Виртом), то можно поступить так. Введем дополнительную переменную b: boolean и напишем:

if k1 < k  then b := (x[k1+1]=t)  else  b:=false; 
{b = (k1<k) and (x[k1+1] = t)} 
while  b  do  begin 
| k1:=k1+1; 
| if k1 < k then b := (x[k1+1]=t) else b:=false; 
end;

Можно также сделать иначе:

end else begin {x[k1+1] = y[l1+1]} 
| if k1 + 1 = k then begin 
| | k1 := k1 + 1; 
| | n := n + 1; 
| end else if x[k1+1] = x [k1+2] then begin 
| | k1 := k1 + 1; 
| end else begin 
| | k1 := k1 + 1; 
| | n := n + 1; 
| end; 
end;

Так будет короче, хотя менее симметрично.

Наконец, можно увеличить размер массива в его описании, включив в него фиктивные элементы.

1.2.19. Даны два неубывающих массива x: array[1..k] of integer и y: array[1..l] of integer. Найти число различных элементов среди x[1],\ldots,x[k],y[1],\ldots,y[l]. (Число действий порядка k+l.)

Татьяна Новикова
Татьяна Новикова
Россия, Пошатово
Artem Bardakov
Artem Bardakov
Россия