Тема 20. Теория игр

20. Теория игр

Вспоминай формулы по каждой теме
Решай новые задачи каждый день
Вдумчиво разбирай решения
ШКОЛКОВО.
Готовиться с нами - ЛЕГКО!
Решаем задачи

Ошибка.
Попробуйте повторить позже

Задача 21#19205Максимум баллов за задание: 1

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

Показать ответ и решение

Решение БУ

from functools import lru_cache
@lru_cache(None)
def game(first_heap, second_heap):  # Функция игры
    if first_heap + second_heap >= 47:  # Если камней в куче стало больше 46
        return 0  # Прекращаем игру
    moves = [game(first_heap + 2, second_heap), game(first_heap, second_heap + 2), game(first_heap * 2, second_heap),
             game(first_heap, second_heap * 2), game(first_heap + 1, second_heap), game(first_heap, second_heap + 1)]  # Генерация всех возможных ходов
    petya_win = [i for i in moves if i <= 0]
    if petya_win: # если в данной позиции есть выигрыш Пети
        return -max(petya_win) + 1
    else: # если в данной позиции выигрыш Вани
        return -max(moves)

for i in range(1,40):
    if game(7,i) == 2: # если в данной позиции возможен выигрыш Пети вторым ходом
        print(i)
        break

Решение АР

from functools import lru_cache

def moves(heap):
    a, b = heap
    return (a + 1, b), (a, b + 1),
           (a + 2, b), (a, b + 2),
           (a * 2, b), (a, b * 2)

@lru_cache(None)
def game(heap):
    if sum(heap) >= 47:
        return ’END’
    elif any(game(x) == ’END’ for x in moves(heap)):
        return ’P1’
    elif all(game(x) == ’P1’ for x in moves(heap)):
        return ’V1’
    elif any(game(x) == ’V1’ for x in moves(heap)):
        return ’P2’
    elif all(game(x) == ’P1’ or game(x) == ’P2’ for x in moves(heap)):
        return ’V2’

for s in range(1, 40):
    print(s, game((7, s)))

Ответ: 16

Ошибка.
Попробуйте повторить позже

Задача 22#19951Максимум баллов за задание: 1

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

Показать ответ и решение
from functools import lru_cache


@lru_cache(None)
def f(a, b):
    if (a + b) <= 12:
        return 0
    moves = []
    if a - 2 >= 0:
        moves.append(f(a - 2, b))
    if b - 2 >= 0:
        moves.append(f(a, b - 2))
    if a > 0:
        moves.append(f(a // 2, b))
    if b > 0:
        moves.append(f(a, b // 2))
    petya_win = [i for i in moves if i <= 0]
    if petya_win:
        return -max(petya_win) + 1
    return -max(moves)


for i in range(1, 100):
    if f(25, i) == 2:
        print(i)
        break

Ответ: 2

Ошибка.
Попробуйте повторить позже

Задача 23#19954Максимум баллов за задание: 1

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

Показать ответ и решение
from functools import lru_cache


@lru_cache(None)
def f(a, b):
    if (a + b) <= 75:
        return 0
    moves = []
    if a - 3 >= 0:
        moves.append(f(a - 3, b))
    if b - 3 >= 0:
        moves.append(f(a, b - 3))
    if a > 0:
        moves.append(f(a // 5, b))
    if b > 0:
        moves.append(f(a, b // 5))
    petya_win = [i for i in moves if i <= 0]
    if petya_win:
        return -max(petya_win) + 1
    return -max(moves)


for i in range(50, 101):
    if f(100, i) == 2:
        print(i)

Ответ: 57

Ошибка.
Попробуйте повторить позже

Задача 24#19957Максимум баллов за задание: 1

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

Показать ответ и решение
from functools import lru_cache


@lru_cache(None)
def f(a, b, c):
    if a + b >= 199:
        return 0
    moves = [f(a, b * 2, c)]
    if c == 0:
        moves.append(f(b, a, 1))
    petya_win = [i for i in moves if i <= 0]
    if petya_win:
        return -max(petya_win) + 1
    else:
        return -max(moves)


for i in range(1, 100):
    if f(99, i, 0) == 2:
        print(i)
        break

Ответ: 13

Ошибка.
Попробуйте повторить позже

Задача 25#19960Максимум баллов за задание: 1

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

Показать ответ и решение

Решение БУ

from functools import lru_cache
@lru_cache(None)
def game(first_heap, second_heap):  # Функция игры
    if first_heap + second_heap >= 199:  # Если камней в куче стало больше 198
        return 0  # Прекращаем игру
    moves = [game(first_heap + 1, second_heap), game(first_heap, second_heap + 1), game(first_heap + 2, second_heap),
             game(first_heap, second_heap + 2),
             game(first_heap + first_heap - 1, second_heap), game(first_heap, second_heap + second_heap - 1)]  # Генерация всех возможных ходов
    petya_win = [i for i in moves if i <= 0]
    if petya_win: # если в данной позиции есть выигрыш Пети
        return -max(petya_win) + 1
    else: # если в данной позиции выигрыш Вани
        return -max(moves)

for i in range(2,98):
    if game(97,i) == 2: # если в данной позиции возможен выигрыш Пети вторым ходом
        print(i)
        break

Решение АР

from functools import lru_cache

def moves(h):
    a, b = h
    return (a + 1, b), (a, b + 1), (a + 2, b), (a, b + 2), (a + a - 1, b), (a, b + b - 1)

@lru_cache(None)
def f(h):
    if sum(h) >= 199:
        return "END"
    if any((f(x) == "END") for x in moves(h)):
        return "P1"
    if all((f(x) == "P1") for x in moves(h)):
        return "V1"
    if any((f(x) == "V1") for x in moves(h)):
        return "P2"
    if all((f(x) == "P2" or f(x) == "P1") for x in moves(h)):
        return "V2"

for s in range(97, 1, -1):
    h = 97, s
    if f(h) == "P2":
        print(s, "P2")

Ответ: 3

Ошибка.
Попробуйте повторить позже

Задача 26#19963Максимум баллов за задание: 1

Для игры, описанной в задании #19962, укажите минимальное значение S  , при котором у Пети есть выигрышная стратегия, причём Петя не может выиграть первым ходом, но может выиграть своим вторым ходом независимо от того, как будет ходить Ваня.

Показать ответ и решение

Программное решение

from functools import lru_cache


@lru_cache(None)
def game(first_heap, second_heap):  # Функция игры

    # Если суммарное кол-во камней в кучах стало более 70
    if first_heap + second_heap > 70:
        # Возвращаем 1,
        # которая преобразуется в победу Вани первым ходом из-за "плохого" хода Пети
        return 1

    # Если суммарное кол-во камней в кучах стало не менее 50
    if first_heap + second_heap >= 50:
        return 0  # Прекращаем игру

    # Прописываем возможные ходы в партии
    moves = [
        game(first_heap + 1, second_heap),
        game(first_heap * 2, second_heap),
        game(first_heap, second_heap + 1),
        game(first_heap, second_heap * 2)
    ]

    # Находим значения, через которые может победить Петя
    petya_win = [i for i in moves if i <= 0]

    if petya_win:  # Если такие значения нашлись и список не пуст
        return -max(petya_win) + 1
    else:  # Иначе побеждает Ваня максимальным ходом
        return -max(moves)


for S in range(1, 29 + 1):
    # Если в данной позиции Петя гарантированно выигрывает своим вторым ходом
    if game(20, S) == 2:
        print(S)
        break  # Первое выведенное значение будет минимальным

Ответ: 7

Ошибка.
Попробуйте повторить позже

Задача 27#19966Максимум баллов за задание: 1

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

Показать ответ и решение
from functools import lru_cache


@lru_cache(None)
def f(first_heap, second_heap, third_heap):
    if first_heap + second_heap + third_heap >= 70:
        return 0
    moves = [f(first_heap + 1, second_heap, third_heap), f(first_heap, second_heap + 1, third_heap),
             f(first_heap, second_heap, third_heap + 1),
             f(first_heap * 2, second_heap, third_heap), f(first_heap, second_heap * 2, third_heap),
             f(first_heap, second_heap, third_heap * 2)]
    petya_win = [i for i in moves if i <= 0]
    if petya_win:
        return -max(petya_win) + 1
    else:
        return -max(moves)


for i in range(1, 40):
    if f(20, 10, i) == 2:
        print(i)
        break

Ответ: 9

Ошибка.
Попробуйте повторить позже

Задача 28#19969Максимум баллов за задание: 1

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

Показать ответ и решение
from functools import lru_cache


@lru_cache(None)
def f(first_heap, second_heap, third_heap, fourth_heap):
    if first_heap + second_heap + third_heap + fourth_heap >= 70:
        return 0
    moves = [f(first_heap + 2, second_heap, third_heap, fourth_heap),
             f(first_heap, second_heap + 2, third_heap, fourth_heap),
             f(first_heap, second_heap, third_heap + 2, fourth_heap),
             f(first_heap, second_heap, third_heap, fourth_heap + 2),
             f(first_heap * 3, second_heap, third_heap, fourth_heap),
             f(first_heap, second_heap * 3, third_heap, fourth_heap),
             f(first_heap, second_heap, third_heap * 3, fourth_heap),
             f(first_heap, second_heap, third_heap, fourth_heap * 3)]
    petya_win = [i for i in moves if i <= 0]
    if petya_win:
        return -max(petya_win) + 1
    else:
        return -max(moves)


for i in range(1, 40):
    if f(9, 6, 8, i) == 2:
        print(i)
        break

Ответ: 5

Ошибка.
Попробуйте повторить позже

Задача 29#19972Максимум баллов за задание: 1

Для игры, описанной в задании #19971, укажите минимальное значение S  , при котором у Пети есть выигрышная стратегия, причём Петя не может выиграть первым ходом, но может выиграть своим вторым ходом независимо от того, как будет ходить Ваня.

Показать ответ и решение

Решение БУ

from functools import lru_cache


@lru_cache(None)
def game(heap_1, heap_2, heap_3, heap_4):  # Функция игры

    # Если суммарное кол-во камней в кучах стало более 80
    if heap_1 + heap_2 + heap_3 + heap_4 > 80:
        # Возвращаем 1,
        # которая преобразуется в победу Вани первым ходом из-за "плохого" хода Пети
        return 1

    # Если суммарное кол-во камней в кучах стало не менее 60
    if heap_1 + heap_2 + heap_3 + heap_4 >= 60:
        return 0  # Прекращаем игру

    # Прописываем возможные ходы в партии
    moves = [
        game(heap_1 + 2, heap_2, heap_3, heap_4), game(heap_1 * 3, heap_2, heap_3, heap_4),
        game(heap_1, heap_2 + 2, heap_3, heap_4), game(heap_1, heap_2 * 3, heap_3, heap_4),
        game(heap_1, heap_2, heap_3 + 2, heap_4), game(heap_1, heap_2, heap_3 * 3, heap_4),
        game(heap_1, heap_2, heap_3, heap_4 + 2), game(heap_1, heap_2, heap_3, heap_4 * 3)
    ]

    # Находим значения, через которые может победить Петя
    petya_win = [i for i in moves if i <= 0]

    if petya_win:  # Если такие значения нашлись и список не пуст
        return -max(petya_win) + 1
    else:  # Иначе побеждает Ваня максимальным ходом
        return -max(moves)


for S in range(1, 29 + 1):
    # Если в данной позиции Петя гарантированно выигрывает своим вторым ходом
    if game(10, 10, 10, S) == 2:
        print(S)
        break  # Первое выведенное значение будет минимальным

Решение АР

from functools import lru_cache

def moves(h):
    a, b, c, d = h
    return (a + 2, b, c, d), (a, b + 2, c, d), (a, b, c + 2, d),
    (a, b, c, d + 2), (a * 3, b, c, d), (a, b * 3, c, d),
    (a, b, c * 3, d), (a, b, c, d * 3)

@lru_cache(None)
def f(h):
    if sum(h) > 80:
        return "V0"
    if sum(h) >= 60:
        return "END"
    if any((f(x) == "END") for x in moves(h)):
        return "P1"
    if all((f(x) == "P1") for x in moves(h)):
        return "V1"
    if any((f(x) == "V1") for x in moves(h)):
        return "P2"
    if all((f(x) == "P2" or f(x) == "P1" or f(x) == "V0") for x in moves(h)):
        return "V2"

for s in range(29, 0, -1):
    h = 10, 10, 10, s
    if f(h) == "P2":
        print(s, "P2")

Ответ: 2

Ошибка.
Попробуйте повторить позже

Задача 30#19975Максимум баллов за задание: 1

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

Показать ответ и решение
from functools import lru_cache


@lru_cache(None)
def f(x_coord, y_coord):
    if (x_coord ** 2 + y_coord ** 2) ** 0.5 > 20:
        return 0
    moves = [f(x_coord + 1, y_coord + 2), f(x_coord, y_coord + 1)]
    petya_win = [i for i in moves if i <= 0]
    if petya_win:
        return -max(petya_win) + 1
    else:
        return -max(moves)


for i in range(1, 18):
    if f(10, i) == 2:
        print(i)
        break

Ответ: 12

Ошибка.
Попробуйте повторить позже

Задача 31#19978Максимум баллов за задание: 1

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

Показать ответ и решение
from functools import lru_cache


@lru_cache(None)
def f(x_coord, y_coord):
    if (x_coord ** 2 + y_coord ** 2) ** 0.5 > 30:
        return 0
    moves = [f(x_coord + 1, y_coord + 2), f(x_coord, y_coord + 1), f(x_coord + 10, y_coord + 1)]
    petya_win = [i for i in moves if i <= 0]
    if petya_win:
        return -max(petya_win) + 1
    else:
        return -max(moves)


for i in range(1, 26):
    if f(15, i) == 2:
        print(i)
        break

Ответ: 11

Ошибка.
Попробуйте повторить позже

Задача 32#22900Максимум баллов за задание: 1

Известно, что Ваня выиграл своим первым ходом после неудачного первого хода Пети (игра описана в предыдущем задании).

Укажите минимальное значение S  , когда такая ситуация возможна.

Показать ответ и решение

Решение БУ

from functools import lru_cache
@lru_cache(None)
def game(first_heap, second_heap):  # Функция игры
    if first_heap + second_heap >= 43:  # Если камней в куче стало больше 42
        return 0  # Прекращаем игру
    moves = [game(first_heap + 2, second_heap), game(first_heap, second_heap + 2), game(first_heap * 2, second_heap),
             game(first_heap, second_heap * 2)]  # Генерация всех возможных ходов
    petya_win = [i for i in moves if i <= 0]
    if petya_win: # если в данной позиции есть выигрыш Пети
        return -max(petya_win) + 1
    else: # если в данной позиции выигрыш Вани
        return -max(moves)

for i in range(1,35):
    if game(8+2,i) == 1 or game(8*2,i) == 1 or game(8,i+2) == 1 or game(8,i*2) == 1: # если в данной позиции после неудачного хода Пети возможен выигрыш Вани
        print(i)
        break

Решение АР

from functools import lru_cache


def moves(h):
    a, b = h
    return (a+2, b), (a, b+2), (a*2, b), (a, b*2)


@lru_cache(None)
def f(h):
    if sum(h) >= 43:
        return ’END’
    if any(f(x) == ’END’ for x in moves(h)):
        return ’WIN1’
    if any(f(x) == ’WIN1’ for x in moves(h)):
        return ’LOSE1’


for i in range(1, 35):
    h = 8, i
    if f(h) == ’LOSE1’:
        print(i)
        break

Ответ: 9

Ошибка.
Попробуйте повторить позже

Задача 33#23507Максимум баллов за задание: 1

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

Показать ответ и решение

Решение БУ

from functools import lru_cache
@lru_cache(None)
def game(first_heap, second_heap):  # Функция игры
    if first_heap + second_heap >= 57:  # Если камней в куче стало больше 56
        return 0  # Прекращаем игру
    moves = [game(first_heap + 1, second_heap), game(first_heap, second_heap + 1), game(first_heap * 2, second_heap),
             game(first_heap, second_heap * 2)]  # Генерация всех возможных ходов
    petya_win = [i for i in moves if i <= 0]
    if petya_win: # если в данной позиции есть выигрыш Пети
        return -max(petya_win) + 1
    else: # если в данной позиции выигрыш Вани
        return -max(moves)

for i in range(1,52):
    if game(5,i) == 2: # если в данной позиции возможен выигрыш Пети вторым ходом
        print(i)

Решение АР

from functools import lru_cache


def moves(h):
    a, b = h
    return (a + 1, b), (a, b + 1), (a * 2, b), (a, b * 2)


@lru_cache(None)
def f(h):
    if (sum(h) >= 57):
        return ’END’
    if any(f(x) == ’END’ for x in moves(h)):
        return ’P1’
    if all(f(x) == ’P1’ for x in moves(h)):
        return ’V1’
    if any(f(x) == ’V1’ for x in moves(h)):
        return ’P2’


for s in range(1, 52):
    h = 5, s
    if f(h) == ’P2’:
        print(s, end=’’)

Ответ: 2325

Ошибка.
Попробуйте повторить позже

Задача 34#23622Максимум баллов за задание: 1

Укажите два максимальных значения S  , при котором у Пети есть выигрышная стратегия, причём:

– Петя не может выиграть за один ход

– Петя может выиграть своим вторым ходом независимо от того, как будет ходить Ваня.

В ответе запишите числа в порядке возрастания без пробелов и знаков препинаний.

Показать ответ и решение

Решение руками

Из предыдущей задачи мы знаем, что в отрезке [61;70] Петя гарантированно выигрывает своим первым ходом. В значении 60 Ваня гарантированно выигрывает своим первым ходом после любого хода Пети. Значит, что если после первого хода Пети в куче станет 60 камней, значит, он гарантированно выиграет в партии своим вторым ходом. Распишем из каких значений за один ход можно попасть в 60:

Если S = 59  , то Пете достаточно прибавить к куче 1  камень. Тогда Ваня сможет увеличить количество камней либо до 61  , либо до 70  , откуда Пете достаточно прибавить 10  к любому из этих значений.

Если S = 50  , то Пете достаточно прибавить к куче 10  камней. Далее аналогично предыдущим действиям.

Решение БУ

from functools import lru_cache
@lru_cache(None)
def game(first_heap): # функция игры
    if first_heap >= 71: # если камней в куче стало больше 70
        return 0 # прекращаем игру
    moves = [game(first_heap+1),game(first_heap+10)] # прописываем ходы возможные в партии
    petya_win = [i for i in moves if i <= 0]
    if petya_win: # проверяем есть ли выигрыш Пети в данной позиции
        return -max(petya_win) + 1
    else: # если в данной позиции выигрыш Вани
        return -max(moves)

for i in range(1,71):
    if game(i) == 2: # если в данной позиции Петя побеждает вторым ходом
        print(i)

Ответ: 5059

Ошибка.
Попробуйте повторить позже

Задача 35#23625Максимум баллов за задание: 1

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

Показать ответ и решение

Решение 1

Возможные значения S : 82,81,61,43,42  . В каждом из этих случаев Петя не может выиграть первым ходом. При S = 82  Петя может получить позицию (20,41)  , при S = 42  — позицию (20,21)  .

В первом случае после хода Вани возникнет одна из позиций (19,41)  , (20,40)  , (10,41)  , (20,21)  , во втором случае — одна из позиций (19,21)  , (20,20)  , (10,21)  , (20,11)  . В любой из перечисленных позиций Петя может выиграть, уменьшив вдвое количество камней в большей куче.

 

Решение 2

from functools import lru_cache


@lru_cache(None)
def f(a, b):
    if a + b <= 40:
        return 0
    moves = []
    if a > 0:
        moves.append(f(a - 1, b))
        if a > 1:
            moves.append(f(a // 2 + (a % 2), b))
    if b > 0:
        moves.append(f(a, b - 1))
        if b > 1:
            moves.append(f(a, b // 2 + (b % 2)))
    petya_win = [i for i in moves if i <= 0]
    if petya_win:
        return -max(petya_win) + 1
    else:
        return -max(moves)


for i in range(20, 100):
    if f(20, i) == 2:
        print(i)

Ответ: 4282

Ошибка.
Попробуйте повторить позже

Задача 36#23628Максимум баллов за задание: 1

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

В ответе запишите числа в порядке возрастания без пробелов и знаков препинаний.

Показать ответ и решение

Решение руками

Из предыдущего задания мы знаем, что в отрезке [20;38] Петя побеждает гарантированно своим первым ходом. Единственное значение, из которого гарантированно можно попасть в область [20;38] – это 19  . Это значение, в котором Ваня побеждает своим первым ходом. Если после хода Пети в куче будет значение 19, значит, он гарантированно победит в данной партии своим вторым ходом. Распишем подробно из каких значений и при каких стратегиях можно попасть в 19  :

Возможные значения S : 17,18.  В этих случаях Петя не может выиграть первым ходом. Однако он может получить кучу из 19  камней (при S = 17  он добавляет 2  камня; при S = 18  1  камень). Тогда после первого хода Ваня в куче будет 20  , 21  или 38  камней. Во всех случаях Петя увеличивает количество камней в 2  раза и выигрывает в один ход.

Решение БУ

from functools import lru_cache
@lru_cache(None)
def game(first_heap): # функция игры
    if first_heap >= 39: # если камней в куче стало больше 38
        return 0 # прекращаем игру
    moves = [game(first_heap+1),game(first_heap+2),game(first_heap*2)] # прописываем ходы возможные в партии
    petya_win = [i for i in moves if i <= 0]
    if petya_win: # проверяем есть ли выигрыш Пети в данной позиции
        return -max(petya_win) + 1
    else: # если в данной позиции выигрыш Вани
        return -max(moves)

for i in range(1,38):
    if game(i) == 2: # если в данной позиции Петя побеждает вторым ходом
        print(i)

Ответ: 1718

Ошибка.
Попробуйте повторить позже

Задача 37#23631Максимум баллов за задание: 1

Для игры, описанной в предыдущем задании, найдите два таких максимальных значения S  , при которых у Пети есть выигрышная стратегия, причём одновременно выполняются два условия:

– Петя не может выиграть за один ход;

– Петя может выиграть своим вторым ходом независимо от того, как будет ходить Ваня.

В ответе запишите числа в порядке возрастания без пробелов и знаков препинаний.

Показать ответ и решение

Решение руками

Из предыдущей задачи мы знаем, что при значениях 17  и 18  Ваня гарантированно победит своим первым ходом. Если после первого хода Пети в куче будет 17  или 18  камней, значит, в этой позиции Петя гарантированно победит вторым ходом. Распишем значения и стратегии, при которых Петя победит своим вторым ходом: При S = 6  Петя может перейти в позицию 18  ходом ∗3  и гарантированно после хода Вани выиграет партию своим вторым ходом.

При S = 13  Петя может перейти в позицию 17  ходом + 4  и гарантированно после хода Вани выиграет партию своим вторым ходом.

При S = 14  Петя может перейти в позицию 18  ходом + 4  и гарантированно после хода Вани выиграет партию своим вторым ходом.

При S = 15  Петя может перейти в позицию 17  ходом + 2  и гарантированно после хода Вани выиграет партию своим вторым ходом.

При S = 16  Петя может перейти в позицию 18  ходом + 2  и гарантированно после хода Вани выиграет партию своим вторым ходом.

В ответ нужно указать два максимальных значениях S. Ответ: 1516

Решение БУ

from functools import lru_cache
@lru_cache(None)
def game(first_heap): # функция игры
    if first_heap >= 55: # если камней в куче стало больше 54
        return 0 # прекращаем игру
    moves = [game(first_heap+2),game(first_heap+4),game(first_heap*3)] # прописываем ходы возможные в партии
    petya_win = [i for i in moves if i <= 0]
    if petya_win: # проверяем есть ли выигрыш Пети в данной позиции
        return -max(petya_win) + 1
    else: # если в данной позиции выигрыш Вани
        return -max(moves)

for i in range(1,55):
    if game(i) == 2: # если в данной позиции Петя побеждает вторым ходом
        print(i)

Ответ: 1516

Ошибка.
Попробуйте повторить позже

Задача 38#23682Максимум баллов за задание: 1

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

Показать ответ и решение

Решение руками

Из предыдущего задания мы знаем, что в отрезке значений [9;26] Петя побеждает своим первым ходом. Для начала нам нужно определить значения, в которых Ваня побеждает своим первым ходом. Значение, из которого ВСЕ первые ходы ведут в вышеописанный отрезок – это значение, в котором Ваня побеждает своим первым ходом. Распишем значения и стратегии, при которых Ваня побеждает своим первым ходом:

S = 7  . Петя может увеличить количество камней до 9  или 21  . В обоих случаях Ване не составит труда завершить партию следующим ходом.

S = 8  . Петя может увеличить количество камней до 10  или 24  . В обоих случаях Ване не составит труда завершить партию следующим ходом.

После того как мы определили значения, при которых Ваня побеждает первым ходом мы можем определить значения, в которых Петя побеждает своим вторым ходом. Значение, из которого ХОТЯ БЫ ОДИН ход ведёт в 7  или 8  – это значение, в котором Петя побеждает своим вторым ходом. Распишем значения и стратегии, при которых Петя побеждает своим вторым ходом:

Нам подходят S = 5;6  . При 5  Петя сходит в 7  , прибавлением двух камней в кучу. Далее Ваня либо добавит   2  камня и станет 9  камней, либо увеличит кол-во камней в куче в 3  раза и станет 21  камень. И из 9  , и из 21  Петя выиграет увеличением кол-ва камней в куче в 3  раза.

В S = 6  Петя может увеличить количество камней до 8  ходом + 2  и далее он гарантированно победит своим вторым ходом.

В ответ нужно указать максимальное значение S. Ответ: 6

Решение БУ

from functools import lru_cache
@lru_cache(None)
def game(first_heap): # функция игры
    if first_heap >= 27: # если камней в куче стало больше 26
        return 0 # прекращаем игру
    moves = [game(first_heap+2),game(first_heap*3)] # прописываем ходы возможные в партии
    petya_win = [i for i in moves if i <= 0]
    if petya_win: # проверяем есть ли выигрыш Пети в данной позиции
        return -max(petya_win) + 1
    else: # если в данной позиции выигрыш Вани
        return -max(moves)

for i in range(1,27):
    if game(i) == 2: # если в данной позиции Петя побеждает вторым ходом
        print(i)

Решение АР

from functools import lru_cache
def moves(h):
    return (h+2), (h*3)
@lru_cache(None)
def game(h):
    if h >= 27:
        return ’END’
    elif any(game(x) == ’END’ for x in moves(h)):
        return ’WIN1’
    elif all(game(x) == ’WIN1’ for x in moves(h)):
        return ’LOSE1’
    elif any(game(x) == ’LOSE1’ for x in moves(h)):
        return ’WIN2’
    elif all(game(x) == ’WIN2’ or game(x) == ’WIN1’ for x in moves(h)):
        return ’LOSE2’
for s in range(1, 27):
    if game(s) == ’WIN2’:
        print(s)

Ответ: 6

Ошибка.
Попробуйте повторить позже

Задача 39#23685Максимум баллов за задание: 1

Для игры, описанной в задании #23684, найдите количество таких значений S,  при которых у Пети есть выигрышная стратегия, причём Петя не может выиграть за один ход и Петя может выиграть своим вторым ходом независимо от того, как будет ходить Ваня.

Показать ответ и решение

Решение руками

Из предыдущего задания мы знаем, что в отрезке значений [11;30] Петя побеждает своим первым ходом. Для начала нужно определить значения, в которых Ваня побеждает своим первым ходом. Значения, из которых ВСЕ первые ходы ведут в вышеописанный отрезок – это значения, в которых Ваня побеждает своим первым ходом. Распишем значения и стратегии, при которых Ваня побеждает своим первым ходом:

S = 9  . Петя может увеличить количество камней до 11  , 18  или 27  камней. Во всех случаях Ване не составит труда следующим ходом завершить партию.

S = 10  . Петя может увеличить количество камней до 12  , 20  или 30  камней. Во всех случаях Ване не составит труда следующим ходом завершить партию.

После того как мы определили значения, в которых Ваня побеждает своим первым ходом мы можем определить значения, в которых Петя побеждает своим вторым ходом. Значение, из которого ХОТЯ БЫ ОДИН ход ведёт в 9  или       10  – это значение, в котором Петя побеждает своим вторым ходом. Распишем значения и стратегии, при которых Петя победит вторым ходом:

S = 3  . Петя может увеличить количество камней до 9  ходом ∗3

S = 5  . Петя может увеличить количество камней до 10  ходом ∗2

S = 7  . Петя может увеличить количество камней до 9  ходом + 2

S = 8  . Петя может увеличить количество камней до 10  ходом + 2

В ответ нужно указать количество значений S. Ответ: 4

Решение программой

from functools import lru_cache
@lru_cache(None)
def game(first_heap): # функция игры
    if first_heap >= 31: # если камней в куче стало больше 30
        return 0 # прекращаем игру
    moves = [game(first_heap+2),game(first_heap*3),game(first_heap*2)] # прописываем ходы возможные в партии
    petya_win = [i for i in moves if i <= 0]
    if petya_win: # проверяем есть ли выигрыш Пети в данной позиции
        return -max(petya_win) + 1
    else: # если в данной позиции выигрыш Вани
        return -max(moves)
count = 0
for i in range(1,31):
    if game(i) == 2: # если в данной позиции Петя побеждает вторым ходом
        count += 1
print(count)

Ответ: 4

Ошибка.
Попробуйте повторить позже

Задача 40#23688Максимум баллов за задание: 1

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

Показать ответ и решение

Решение БУ

from functools import lru_cache
@lru_cache(None)
def game(first_heap, second_heap):  # Функция игры
    if first_heap + second_heap >= 66:  # Если камней в куче стало больше 65
        return 0  # Прекращаем игру
    moves = [game(first_heap + 1, second_heap), game(first_heap, second_heap + 1), game(first_heap * 2, second_heap),
             game(first_heap, second_heap * 2)]  # Генерация всех возможных ходов
    petya_win = [i for i in moves if i <= 0]
    if petya_win: # если в данной позиции есть выигрыш Пети
        return -max(petya_win) + 1
    else: # если в данной позиции выигрыш Вани
        return -max(moves)

for i in range(1,55):
    # если в данной позиции возможен выигрыш Пети вторым ходом
    if game(11,i) == 2:
        print(i)

Ответ: 26
Рулетка
Вы можете получить скидку в рулетке!