Программирование с ограничениями (CP) - это парадигма для решения комбинаторных задач, охватывающая широкий спектр методов из искусственного интеллекта, информатики и исследования операций. В программировании с ограничениями пользователи декларативно устанавливают ограничения на возможные решения для набора переменных решения. Ограничения отличаются от обычных примитивов языков императивного программирования тем, что они не определяют шаг или последовательность шагов для выполнения, а скорее свойства решения, которое необходимо найти. Помимо ограничений, пользователям также необходимо указать метод решения этих ограничений. Обычно это основано на стандартных методах, таких как хронологический поиск с возвратом и распространение ограничений, но может использоваться настраиваемый код, например, специфическая проблема ветвления эвристика.
Программирование ограничений берет свое начало и может быть выражено в форме логического программирования ограничений, которое встраивает ограничения в логическую программу . Этот вариант логического программирования принадлежит Яффару и Лассезу, которые расширили в 1987 году определенный класс ограничений, которые были введены в Prolog II. Первыми реализациями программирования логики ограничений были CLP (R) и CHIP.
Вместо логического программирования ограничения могут быть смешаны с функциональным программированием, переписывание терминов и императивные языки. Языки программирования со встроенной поддержкой ограничений включают Oz (функциональное программирование) и Kaleidoscope (императивное программирование). В основном ограничения реализуются в императивных языках с помощью наборов инструментов для решения ограничений, которые представляют собой отдельные библиотеки для существующего императивного языка.
Программирование с ограничениями - это встраивание ограничений в основной язык. Первыми используемыми хост-языками были языки логического программирования, поэтому изначально это поле называлось программированием логики ограничений. Эти две парадигмы имеют много общих важных функций, таких как логические переменные и возврат. Сегодня большинство реализаций Prolog включают одну или несколько библиотек для программирования логики ограничений.
Разница между ними во многом заключается в их стилях и подходах к моделированию мира. Некоторые задачи более естественно (и, следовательно, проще) писать как логические программы, тогда как некоторые более естественно писать как программы с ограничениями.
Подход программирования ограничений заключается в поиске состояния мира, в котором одновременно выполняется большое количество ограничений. Проблема обычно определяется как состояние мира, содержащее ряд неизвестных переменных. Программа ограничения ищет значения для всех переменных.
Временное параллельное программирование с ограничениями (TCC) и недетерминированное временное параллельное программирование с ограничениями (MJV) - это варианты программирования с ограничениями, которые могут иметь дело со временем.
Ограничение - это отношение между несколькими переменными, которое ограничивает значения, которые эти переменные могут принимать одновременно.
Определение - проблема удовлетворения ограничений в конечных областях (или CSP) определяется триплетом где:
Существуют три категории ограничений:
Определение - назначение (или модель) CSP определяется пара где:
Присваивание связь переменной со значением из ее домена. Частичное присвоение - это когда было присвоено подмножество переменных проблемы. Полное присвоение - это когда все переменные проблемы были присвоены.
Свойство - Учитывая назначение (частичное или полное) CSP и ограничение например, , присвоение удовлетворяет ограничению тогда и только тогда, когда все значения из переменные ограничения принадлежит .
Определение - Решение CSP - это полное назначение, которое удовлетворяет всем ограничениям задачи.
Во время поиска решений CSP пользователь может пожелать:
Задача оптимизации ограничений (COP) - это проблема удовлетворения ограничений, связанная с целевой функцией.
Оптимальное решение для минимизации (максимизации) COP - это решение, которое минимизирует (максимизирует) значение целевой функции.
Во время поиска решений CSP пользователь может пожелать:
Языки для программирования на основе ограничений используют один из двух подходов :
Распространение ограничений в проблемы удовлетворения ограничений является типичным примером уточняющей модели, а электронные таблицы являются типичным примером модели возмущения.
Модель уточнения является более общей, поскольку она не ограничивает переменные одним значением, она может привести к нескольким решениям одной и той же проблемы. Однако модель возмущений более интуитивно понятна для программистов, использующих объектно-ориентированные языки со смешанными императивными ограничениями.
Ограничения, используемые в программировании ограничений, обычно относятся к некоторым конкретным областям. Некоторые популярные области программирования с ограничениями:
Конечные домены - одна из наиболее успешных областей программирования с ограничениями. в некоторых областях (например, исследование операций ) программирование ограничений часто отождествляется с программированием ограничений в конечных областях.
Локальная согласованность условия являются свойствами удовлетворения ограничений проблемы, связанные с согласованностью подмножеств переменных или ограничений. Их можно использовать для уменьшения пространства поиска и создания проблему проще решить. Используются различные виды условий локальной согласованности, в том числе согласованность узлов, согласованность дуги и согласованность путей .
Каждое условие локальной согласованности может быть реализовано преобразованием, изменяющим проблема без изменения ее решения. Такое преобразование называется распространением ограничений. Распространение ограничений работает за счет сокращения областей переменных, усиления ограничений или создания новых. Это приводит к сокращению пространства поиска, что упрощает решение проблемы с помощью некоторых алгоритмов. Распространение ограничений также может использоваться как средство проверки на неудовлетворенность, неполное в целом, но полное в некоторых частных случаях.
Существует три основных алгоритмических метода решения проблем удовлетворения ограничений: поиск с возвратом, локальный поиск и динамическое программирование.
Поиск с возвратом - это общий алгоритм для поиска всех (или некоторых) решений некоторых вычислительных проблем, в частности проблем удовлетворения ограничений, который постепенно создает кандидатов для решений, и отказывается от кандидата («возвращается»), как только определяет, что кандидат не может быть завершен до действительного решения.
Локальный поиск - неполный метод поиска решения проблемы. Он основан на итеративном улучшении присвоения переменных до тех пор, пока не будут выполнены все ограничения. В частности, алгоритмы локального поиска обычно изменяют значение переменной в назначении на каждом этапе. Новое назначение близко к предыдущему в пространстве назначения, отсюда и название локальный поиск.
Динамическое программирование - это одновременно метод математической оптимизации и метод компьютерного программирования. Это относится к упрощению сложной проблемы путем разбивки ее на более простые подзадачи рекурсивным способом . Хотя некоторые проблемы с решениями нельзя разделить таким образом, решения, которые охватывают несколько моментов времени, часто рекурсивно распадаются. Точно так же в информатике, если проблема может быть решена оптимально, разбив ее на подзадачи, а затем рекурсивно найдя оптимальные решения подзадач, то говорят, что она имеет оптимальную подструктуру.
Синтаксис для выражения ограничений для конечных доменов зависит от основного языка. Ниже приведена программа Prolog, которая решает классическую буквенную головоломку SEND + MORE = MONEY в программировании логики ограничений:
% Этот код работает как в YAP и SWI-Prolog, использующий поставляемую средой библиотеку решателя ограничений% CLPFD. Для работы в других средах Prolog или с использованием других средств решения ограничений могут потребоваться незначительные изменения. : - use_module (библиотека (clpfd)). sendmore (Digits): - Digits = [S, E, N, D, M, O, R, N, Y],% Создание переменных Digits ins 0..9,% Связывание доменов с переменными S # \ = 0,% Ограничение: S должно отличаться от 0 M # \ = 0, all_different (Digits),% все элементы должны принимать разные значения 1000 * S + 100 * E + 10 * N + D% Другие ограничения + 1000 * M + 100 * O + 10 * R + E # = 10000 * M + 1000 * O + 100 * N + 10 * E + Y, метка (цифры). % Начать поиск
Интерпретатор создает переменную для каждой буквы в головоломке. Оператор ins
используется для указания доменов этих переменных, чтобы они находились в наборе значений {0,1,2,3,..., 9}. Ограничения S # \ = 0
и M # \ = 0
означают, что эти две переменные не могут принимать нулевое значение. Когда интерпретатор оценивает эти ограничения, он сокращает области этих двух переменных, удаляя из них значение 0. Затем рассматривается ограничение all_different (Digits)
; он не уменьшает ни один домен, поэтому просто сохраняется. Последнее ограничение указывает, что цифры, присвоенные буквам, должны быть такими, чтобы "SEND + MORE = MONEY" сохранялось, когда каждая буква заменяется соответствующей ей цифрой. Из этого ограничения решатель делает вывод, что M = 1. Все сохраненные ограничения, включающие переменную M, пробуждаются: в этом случае распространение ограничения на ограничении all_different
удаляет значение 1 из домена всех оставшихся переменных. Распространение ограничений может решить проблему путем сведения всех доменов к одному значению, оно может доказать, что проблема не имеет решения, уменьшив область до пустого набора, но также может завершиться без доказательства выполнимости или неудовлетворенности. Литералы label используются для фактического поиска решения.
На Викискладе есть материалы, связанные с программированием в ограничениях. |