Привет, абстрагирующимся. Прочитав эту статью, задумался,
а как представлять эту задачу языком Пролог? Попробую выразить
свое, затянувшееся, субботнее отношение к этой пятничной задаче, с
помощью доступных декларативных формулировок.
В реализации на Скала, я увидел операцию "(value % n)" и пояснение,
что значения value,n -это: type class "Integral" требующий от
типа "T" возможности вычислять остаток от деления и иметь значение
"zero".
Это меня подтолкнуло, на такую мысль, а может абстрагируемся еще
больше, и отбросим арифметические операции этого
"интэграл", может рассмотрим глубже идею натуральных
чисел, сейчас попробую продемонстрировать и получить
реализацию...
Это представляли еще в Древней Греции, натуральное число N это 0
или следующее за ним натуральное число, результат функции p(N).
Записывать можно так N0= 0,
потом N1=p(0), и далее N2=p(p(0)) вот такой ряд натуральных чисел,
от 0 и далее следующее и следующее значение:
Изображаю это на Прологе:
natural(0).natural(N+1):-natural(N).
если выполнить такую цель, получим бесконечность ответов:
?-natural(X).X=0;X=0+1;X=0+1+1: .... ну и тд.
От этого, пользы большой не извлечь, а как часть объяснения
годится, значения переменной X являются структурами, в Прологе есть
инфиксные операции, которые могут соединить значения между собой,
вместо громоздкой записи p(p(p..(p(0)..))), просто соединим символы
знаком "+": 0+1+1+1+1 это просто такое символьное выражение, почти
строка, а можно бы записать и как: zero plus one plus one.
Вот так, можно объявлять свой знак операции:
:-op(600, yfx, plus). %op(+Precedence, +Type, :Name) natural(zero).natural(N plus one):-natural(N).
Будут такие представления чисел:
?- natural(X).X = zero ;X = zero plus one ;X = zero plus one plus one ; ...
Далее, полезно будет сформулировать следующую проблему, как
выражать сложение или вычитание над двумя такими натуральными
числами?,
вот оно:
add(0, X, X).add(X+1, Y, Z+1):- add(X, Y, Z).
Теперь можно решить вот такие цели для сложения:
?- add(0+1, 0+1+1, X).X=0+1+1+1
И вот такой вопрос, превратится в вычитание натуральных чисел:
?- add(X, 0+1+1+1, 0+1+1+1+1).X=0+1
Можно получать все возможные натуральные слагаемые, составившие определенное значение:
?- add(X, Y, 0+1+1+1+1).X=0, Y=0+1+1+1+1;X=0+1, Y=0+1+1+1;X=0+1+1, Y=0+1+1;.. и тд.
Подбираемся к сути этой записки, умение абстрагироваться, разделять понятия на более примитивные, до состояния их точного и краткого описания это и есть важная часть программирования, это, думаю, радость любой инжинерии
В задаче, на вход поступят числа, записанные как целые, арабские, и если не отменять это условие и не просить пользователя передавать на вход, только что показанные структуры, реализую такое правило:
int_natural(0,0).int_natural(X, Y+1):- X0 is X-1, int_natural(X0, Y).
оно сможет конвертировать целое число в его "натуральное" представление:
?- int_natural(6,X).X=0+1+1+1+1+1+1
Операция is это встроенное вычисление арифметических
действий.
Теперь ближе к задаче, там было, про определить остаток от деления.
Сначала продемонстрирую умножение:
mul(0, X, 0).mul(X+1, Y, Z):- mul(X, Y, Z1), add(Z1, Y, Z).
это можно использовать и для обратной операции для деления:
?- mul(0+1+1, 0+1+1, X).X=0+1+1+1+1?- mul(0+1+1, X, 0+1+1+1+1).X=0+1+1
Вот так, сформулирую остаток от деления:
rem(X, X, 0).rem(X, Y, R) :- add(X1,Y,X), rem(X1,Y,R),!.rem(X, _, X):-!.
тут и отсечение (!) заранее установленно, что бы не шокировать читателей функцией обратной к "остатку от деления", это, наверное, операция "следующее число, не кратное на одну величину остатка", типа так:
?- rem(X, 0+1+1+1, 0+1).X = 0+1+1+1+1 ;X = 0+1+1+1+1+1+1+1 ;X = ... + ... + 1+1+1+1+1+1+1+1+1 ... и тд.
"Прямое" использование этого правила вот:
?- rem(0+1+1+1+1, 0+1+1, X).X = 0.?- rem(0+1+1+1+1, 0+1+1+1, X).X = 0+1.
FizzBuzz
Отлично, всю задачу сформулировать теперь можно таким видом:
fizz_buzz(N, fizzbuzz) :- rem(N, 0+1+1+1, 0), rem(N, 0+1+1+1+1+1, 0),!.fizz_buzz(N, fizz) :- rem(N, 0+1+1+1, 0),!.fizz_buzz(N, buzz) :- rem(N, 0+1+1+1+1+1, 0),!.fizz_buzz(N, N).
Приходится опять поставить отсечения, для повышения наглядности, с помощью них, не надо думать о противоположных вариантах решений, описываем только истинные варинаты сверху вниз, а последний вариант это исключение из предыдущих. И входом тут является структура, с записью натурального числа, проверим вот так:
?- int_natural(5, N),!, fizz_buzz(N, X).N = 0+1+1+1+1+1,X = buzz.?- int_natural(15, N),!, fizz_buzz(N, X).N = ... + ... + 1+1+1+1+1+1+1+1+1,X = fizzbuzz.
или по условию, на выходе получаем тоже число:
?- int_natural(4, N),!, fizz_buzz(N, X).N = X, X = 0+1+1+1+1.
Исключив предыдущие недостатки и немного подумав про производительность, упрощаю до такого "алгоритма":
fizz_buzz2(IN, R) :-once(int_natural(IN,N)), once(rem(N, 0+1+1+1, 0) -> F=fizz; F=''), once(rem(N, 0+1+1+1+1+1, 0) -> B=buzz; (F\='')->B=''), atom_concat(F,B,R),!.fizz_buzz2(N, N).
В этом варианте сразу переводим входное целое число и компонуем
выходное значение из частей, если это потребовалось. Встроенное
правило once(Goal), выполняет цель только один раз, для
исключения возвратов и зацикливаний.
Вот такие примеры:
?- fizz_buzz2(2, X).X = 2.?- fizz_buzz2(15, X).X = fizzbuzz.
Обработку ряда чисел, можно получить отобразив собрав
все решения, указанной цели в один список, для этого используем
встроенный мета-предикат findall():
?- findall(X, (between(1, 10, N), fizz_buzz2(N, X)), Result).Result = [1, 2, fizz, 4, buzz, fizz, 7, 8, fizz|...].
Вот тут, можно попробовать.
Для заключения
Пролог, это любопытная возможность переносить в "символы" условия задач, это система опирающаяся на логику предикатов и встроенные процедуры автоматического доказательства теорем. Формулировка проблемы, при удачном выражении, позволяет находить решение для самой задачи, а также сразу иметь решение "обратной" к ней проблеме. А может вот таким вопросом, увидим, какие числа, приводят к fizz: ?- fizz_buzz(N, fizz).