Программирование игр и головоломок
Программирование игр и головоломок читать книгу онлайн
Рассматриваются способы программирования различных занимательных игр и головоломок с числами, геометрическими фигурами и др. Изложение большинства игр и головоломок ведется в несколько этапов. Сначала разъясняется сама постановка задачи и требования, предъявляемые к алгоритму ее решения.
В следующем разделе книги обсуждается сам алгоритм и возможные пути его реализации.
В конце книга по многим играм и головоломкам даются наброски их программной реализации. Используемый при этом язык типа Паскаля допускает перевод на другие широко распространенные языки программирования.
Для начинающих программистов, студентов вузов и техникумов.
Внимание! Книга может содержать контент только для совершеннолетних. Для несовершеннолетних чтение данного контента СТРОГО ЗАПРЕЩЕНО! Если в книге присутствует наличие пропаганды ЛГБТ и другого, запрещенного контента - просьба написать на почту [email protected] для удаления материала
<i>s</i> := ЕСЛИ четно (<i>n</i>) ТО 2 ИНАЧЕ 1 КОНЕЦ_ЕСЛИ
<i>d</i> := 0; <i>k</i>:= 2<sup><i>n</i></sup> − 1
ВЫПОЛНЯТЬ
<i>а</i> := <i>d</i> + <i>s</i>; ЕСЛИ <i>a</i> > 2 ТО <i>а</i> := <i>а</i> − 8
КОНЕЦ_ЕСЛИ
переместить диск 1 с <i>d</i> на <i>а</i>;
<i>d</i> : = <i>a</i>; <i>k</i> := <i>k</i> − 1
ЕСЛИ <i>k</i> = 0 TO КОНЧЕНО КОНЕЦ_ЕСЛИ
переместить единственный диск, который можно переместить, кроме диска 1
<i> k</i> := <i>k</i> − 1
ВЕРНУТЬСЯ
Все диски имеют общее свойство: нечетные диски перемещаются в том же направлении, что и диск 1, а четные диски — в другом направлении.
В вышеприведенной программе стратегия совершенна с точки зрения исполнения вручную, потому что в каждый данный момент сразу видно, какой диск нужно переместить, если это не самый маленький диск (меньший из двух остальных дисков перемещается на больший). В нашей программе вам нужно вычислить это движение. Один из наиболее простых способов состоит в том, чтобы представить игру с помощью вектора, дающего для диска i номер стержня, на котором он находится. Диск, подлежащий перемещению — это наименьший Диск, который находится не на том же стержне, что и диск 1, следовательно, номер стержня которого отличается от d. Этот самый диск перемещается со стержня, на котором он находится — с номером x — на стержень 3 − x − d.
Обозначим первое перемещение через 1. Поскольку диск 1 перемещается один раз в каждой паре ходов (точнее, перемещается через ход), то он перемещается в каждый нечетный ход. По индукции покажите, что диск p перемещается в ходы с номерами, которые делятся на 2р−1, но не делятся на 2p (т. е. являются нечетными кратными числа 2p−1).
Номер k любого хода может быть единственным способом представлен в виде
k = (2r + 1)2р-1.
Перемещаемый на этом ходе диск есть диск с номером p, и это — его (r + 1)-е перемещение. Так как он начинает движение со стержня 0 и перемещается в направлении sp (1, если р нечетно, и 2 в противном случае), то на этом ходе диск перемещается с rsp-го на (r + 1)sр-й стержень, где эти числа берутся по модулю 3.
Игра 34.
Попытаемся охарактеризовать значение р, дающее игре оптимум для данного n. Нам известно, что f3(n − p)= 2n-p − 1.
Должно выполняться
2f4(p − 1) + 2n-p+1 − 1 ≥ 2f4(р) + 2n-p − 1,
2f4(p + 1) + 2n-p-1 − 1 ≥ 2f4(р) + 2n-p − 1.
Удобно пользоваться первыми разностями для функции f4:
d(р) = f4(p + 1) − f4(p).
Два приведенных выше соотношения могут быть переписаны следующим образом:
d(p − 1) < 2n-p-1, d(р) ≥ 2n-p-2.
Интересно рассматривать даже не d(р), а скорее 2pd(р) = g(р):
g(р − 1) ~ 2n-2 ≤ g(р).
Можно еще упростить запись, беря не g(р), а величину
h(р) = log2(g(р)) = р + Iog2(d(р)).
Тогда получаем
h(р − 1) < n − 1 ≤ h(р).
При данном n величина р — наименьшее целое, для которого h больше или равно n − 2.
Приведем здесь первые из полученных таким образом значений:
n | q | f 4 | p | d | h |
0 | 0 | 0 | 1 | 0 | |
1 | 1 | 1 | 2 | 2 | |
2 | 3 | 2 | 3 | ||
3 | 2 | 5 | 1 | 4 | 5 |
4 | 9 | 1 | 4 | 6 | |
5 | 13 | 1 | 4 | 7 | |
6 | 3 | 17 | 3 | 8 | 9 |
7 | 25 | 3 | 8 | 10 | |
8 | 33 | 4 | 8 | 11 | |
9 | 41 | 5 | 8 | 12 | |
10 | 4 | 49 | 6 | 16 | 14 |
11 | 65 | 6 | 16 | 15 |
Мы добавили в таблицу переменное q, связанное с «треугольными» числами. Для n = q(q + 1)/2 действительно убеждаемся, что
h(р) = h(р − 1) + 2
в то время как для других n
h(p) = h(p − 1) + 1.
Исходя из n, можно вычислить q:
q = целая_часть ((
− 1)/2).Имеем
h(n) = n + целая_часть ((
− 1)/2).Покажите это по индукции. Исходя отсюда, вычисляется все. Таким образом, если n дано, то р — наименьшее целое, большее или равное
(2n − 1 −
)/2.Игра 35.
Возьмем, например, игру с 50 дисками. Она реализуется переносом сначала 40 дисков на запасной стержень, а затем 10 последних дисков со стержня 0 на стержень 1, с использованием при этом только трех свободных стержней. Наконец, остается перенести начальные 40 самых маленьких дисков с запасного стержня на первый стержень, используя все 4 стержня.
Чтобы переместить 40 дисков с 4 стержнями, сводим задачу к перемещению 31 диска с 4 стержнями, а затем 9 с 3 стержнями…
Таким образом, дело сводится к разбиению 50 дисков на 8 сегментов:
Каждый сегмент перемещается с использованием 3 стержней, в чем мы следуем итеративной стратегии, которая уже описана выше. Единственный вопрос — это правильный выбор запасных стержней.
Договоримся работать с тремя стержнями 0, 1, 2, так что стержень 3 остается пустым и служит запасным стержнем при любом перемещении какого-либо сегмента. Более точно, перемещение сегмента р со стержня d на стержень а осуществляется с помощью изученной выше процедуры Н, в которой запасным стержнем является стержень 3.
Сегмент 1 перемещается в каждый из двух ходов подряд (под ходом я понимаю последовательность операций, реализующих процедуру Н), всегда в одном и том же направлении.