У меня есть проблема комбинаторики, которую я хотел бы поставить в OEIS - проблема в том, что у меня недостаточно терминов. Задача этого кода - помочь мне вычислить больше терминов, и победителем станет пользователь, представивший наибольшее количество терминов.
Проблема
Предположим, я даю вам треугольный набор лампочек с длиной стороны :
o
o o
o o o
o o o o
o o o o o
o o o o o o
1 2 ... n
Я собираюсь включить три лампочки, которые образуют «вертикальный» равносторонний треугольник, как в следующем примере:
o
o x
o o o
o o o o
o x o o x
o o o o o o
Прежде чем я включу свет, ваша задача состоит в том, чтобы убрать как можно больше лампочек из массива, не теряя возможности вывести треугольник включенных лампочек. Чтобы было ясно, если лампочка была удалена, она не горит, когда ее положение включено.
Например, если вы удалили следующие лампочки (помеченные .
), вы увидите только два следующих индикатора (помеченных x
), что достаточно однозначно вывести третью (неосвещенную) позицию:
. .
. o . x
. . o . . o
o o o . => o o o .
o o o o . o x o o . <- the third unlit position
o . . . o o o . . . o o
Позвольте a(n)
быть максимальное количество лампочек, которые могут быть удалены без внесения каких-либо неясностей.
пример
С наивным алгоритмом я проверил значения до треугольника с длиной стороны 7, как показано ниже:
.
. . o
. . o o . o
. . . . . o . o o .
. . . . o o o o o . o o . o .
. . . . o o o o . o o o o o . o . o . o o
. . . o o . o o o o . . o o o . . . o o o . o . o o o
a(2) = 3 a(3) = 4 a(4) = 5 a(5) = 7 a(6) = 9 a(7) = 11
счет
Представление, которое вычисляет последовательность [a(2), a(3), ..., a(n)]
для самых больших n побед. Если два представления имеют идентичные последовательности, выигрывает тот, который был опубликован ранее.
Хотя это и не обязательно для представления, для меня было бы поучительно, если вы опубликуете конструкцию получающихся триангулярных массивов, как в примере выше.
источник
Ответы:
Python 3 ,
n=8
Использует решатель CP-SAT Google OR-Tools .
После работы в течение ~ 30 секунд, он выводит следующее:
n=9
a(9)=15
n=8
Как это устроено
Таким образом, вопрос можно перефразировать как проблему SAT, с одним ограничением для каждой пары треугольников.
PS: Я бы очень хотел привести пример для
n=8
, но у меня проблемы с решателем SAT, который, очевидно, хочет оставить решения для себя.источник
Получение решений из программы @ Delfad0r
Я расширил программу @ Delfad0r для вывода решений. Это также дает промежуточные результаты, так что вы получите такой результат:
Это вычисление заняло несколько часов.
Если вы испытываете нетерпение и нажимаете
Ctrl-C
после того, как какое-то неоптимальное решение было найдено, программа покажет это решение. Так что это не займет много времени, чтобы получить это:Вот расширенная программа:
источник
Python 3
Основываясь строго на ответе Делфадора , в основном следует той же логической последовательности, проверяя пары треугольников и проверяя конфигурацию, если она не содержит пар треугольников, которые не проходят эту проверку. Поскольку я не использовал никаких библиотек, кроме itertools и copy, у меня есть полный контроль над сохранением примеров, встречающихся в программе.
Проблема в том, что это не очень эффективно. Он работает очень быстро до
n=5
, но начинает значительно замедляться после этой точки. Наn=6
это занимает около минуты, и гораздо медленнееn=7
. Я предполагаю, что есть много исправлений эффективности, которые могут быть сделаны с этой программой, но это быстро сделанный предварительный набросок хорошего решения с намного большей гибкостью, чтобы проверить внутреннюю работу этого метода. Я буду постепенно работать над этим со временем.источник