Brainfuck
Материал из Википедии — свободной энциклопедии
Brainfuck (англ. brain+fuck) — один из известнейших эзотерических языков программирования, придуман Урбаном Мюллером (Urban Müller) для забавы. Состоит из восьми команд, каждая из которых записывается одним символом. Исходный код программы на Brainfuck представляет собой последовательность символов языка без какого-либо синтаксиса.
Машина, которой управляют команды Brainfuck, состоит из упорядоченного набора ячеек и указателя текущей ячейки, напоминая ленту и головку машины Тьюринга. Кроме того, подразумевается устройство общения с внешним миром (см. команды . и ,).
8 команд языка Brainfuck: | ||
> | перейти к следующей ячейке | |
< | перейти к предыдущей ячейке | |
+ | увеличить значение в текущей ячейке на 1 | |
- | уменьшить значение в текущей ячейке на 1 | |
. | напечатать значение из текущей ячейки | |
, | ввести извне значение и сохранить в текущей ячейке | |
[ | если значение текущей ячейки нуль, перейти вперёд по тексту программы на ячейку, следующую за соответствующей ] (с учётом вложенности) | |
] | если значение текущей ячейки не нуль, перейти назад по тексту программы до соответствующей [ (с учётом вложенности) |
Язык Brainfuck можно описать с помощью эквивалентов языка Си (предполагается, что переменная ptr объявлена как указатель на байт):
команда Brainfuck | Си эквивалент |
---|---|
> |
++ptr; |
< |
--ptr; |
+ |
++*ptr; |
- |
--*ptr; |
. |
putchar(*ptr); |
, |
*ptr=getchar(); |
[ |
while (*ptr) { |
] |
} |
Несмотря на внешнюю примитивность, Brainfuck с бесконечным набором ячеек имеет тьюринговскую полноту, а, следовательно, по потенциальным возможностям не уступает «настоящим» языкам, подобным Си, Паскалю или Java.
Brainfuck подходит для экспериментов по генетическому программированию из-за простоты синтаксиса, и, соответственно, генерации исходного кода.
В «классическом» Brainfuck, описанном Мюллером, размер ячейки — один байт, количество ячеек 30000. В начальном состоянии указатель находится в крайней левой позиции, а все ячейки заполнены нулями. Увеличение/уменьшение значений ячеек происходит по модулю 256. Ввод/вывод также происходит побайтно, с учётом кодировки ASCII (то есть в результате операции ввода (,) символ 1 будет записан в текущую ячейку как число 0x31 (49), а операция вывода (.), совершённая над ячейкой, содержащей 0x41 (65), напечатает латинскую А). В других вариантах языка размер и количество ячеек может быть другим (бо́льшим). Есть версии, где значение ячеек не целочислено (с плавающей точкой).
[править] Пример программы
- Программа на языке Brainfuck, печатающая «Hello World!»:
++++++++++[>+++++++>++++++++++>+++>+<<<<-]>++ .>+.+++++++..+++.>++.<<+++++++++++++++.>.+++. ------.--------.>+.>.
Разбор программы:
Подготовка в памяти (с ячейки 1) массива значений близких к ASCII-кодам символов, которые необходимо вывести (70, 100, 30, 10), через повторение 10 раз приращения ячеек на 7, 10, 3 и 1, соответственно | ||
++++++++++ |
присваивание ячейке 0 (счетчику) значения 10 | |
[ |
повторять, пока значение текущей ячейки (ячейки 0) больше нуля | |
>+++++++ |
приращение ячейки 1 на 7 | |
>++++++++++ |
приращение ячейки 2 на 10 | |
>+++ |
приращение ячейки 3 на 3 | |
>+ |
приращение ячейки 4 на 1 | |
<<<<- |
возврат к ячейке 0 (счетчику), и его уменьшение на 1 | |
] |
вернутся к началу цикла | |
Получение кодов букв и их вывод | ||
>++. |
Вывод «Н». Получение кода «H» (72) из 70 в ячейке 1 и вывод | |
>+. |
Вывод «e». Получение кода «e» (101) из 100 в ячейке 2 и вывод | |
+++++++.. |
Вывод «ll». Получение кода «l» (108) из 101 в ячейке 2 и вывод дважды | |
+++. |
Вывод «o». Получение кода «o» (111) из 108 в ячейке 2 и вывод | |
>++. |
Вывод пробела. Получение кода пробела (32) из 30 в ячейке 3 и вывод | |
<<+++++++++++++++. |
Вывод «W». Получение кода «W» (87) из 72 в ячейке 1 и вывод | |
>. |
Вывод «o». Код «o» (111) уже находится в ячейке 2, просто его выводим | |
+++. |
Вывод «r». Получение кода «r» (114) из 111 в ячейке 2 и вывод | |
------. |
Вывод «l». Получение кода «l» (108) из 114 в ячейке 2 и вывод | |
--------. |
Вывод «d». Получение кода «d» (100) из 108 в ячейке 2 и вывод | |
>+. |
Вывод «!». Получение кода «!» (33) из 32 в ячейке 3 и вывод | |
>. |
Вывод кода перевода строки (10) из ячейки 4 |
В принципе, печать «Hello World!», можно реализовать проще, но программа будет в три с лишним раза больше чем приведенный выше оптимизированный вариант:
+++++++++++++++++++++++++++++++++++++++++++++ +++++++++++++++++++++++++++.+++++++++++++++++ ++++++++++++.+++++++..+++.------------------- --------------------------------------------- ---------------.+++++++++++++++++++++++++++++ ++++++++++++++++++++++++++.++++++++++++++++++ ++++++.+++.------.--------.------------------ --------------------------------------------- ----.-----------------------.
[править] Программирование на языке Brainfuck
Каждый, начинающий программировать на Brainfuck, немедленно сталкивается со следующими проблемами:
- отсутствие операции копирования значения
- отсутствие промежуточной (аккумуляторной) памяти
- отсутствие условных операторов в их привычном виде
- отсутствие привычной арифметики, операций умножения и деления
Эти проблемы могут быть решены.
Обозначим @(k) сдвиг на k ячеек вправо, если k>0, и влево, если k<0 Соответственно, @(k) = >…k раз…> либо <…-k раз…<
zero(): обнуление текущей ячейки: [-]
add(k): прибавление значения ячейки n (текущей) к значению ячейки n+k: [ — @(k) + @(-k) ] при этом значение ячейки n теряется (обнуляется).
copy(k): копирование значения ячейки n (текущей) в ячейку n+k с потерей (обнулением) значения ячейки n: @(k) zero() @(-k) add(k) = @(k) [-] @(-k) [ — @(k) + @(-k) ]
copy(k,t): копирование значения ячейки n (текущей) в ячейку n+k c использованием промежуточной ячейки n+k+t, блогадаря чему значение ячейки n не теряется (сохраняется). @(k) zero() @(t) zero() @(-k-t) [ — @(k) + @(t) + @(-k-t) ] @(k+t) copy (-k-t) = @(k) [-] @(t) [-] @(-k-t) [ — @(k) + @(t) + @(-k-t) ] @(k+t) [ — @(-k-t) + @(k+t) ]
Brainfuck не используется для практического программирования (за исключением работ отдельных энтузиастов), так как создан заведомо искусственным языком, не имеющим естественной среды (все существующие BF-машины являются эмуляторами). Язык преимущественно используется программистами как головоломка и задача для соревнований.
[править] Ссылки
- Оригинальное описание BF на английском языке и ссылки на BF-ресурсы
- Brainfuck interpreter with integrated debugger (IDE) for Windows
- ru_brainfucker — русское ЖЖ-сообщество любителей эзотерических языков
- статья на rsdn.ru про эзотерические языки программирования
- Processing_BF - оптимизирующий интерпретатор и транслятор в PHP, написанный на языке PHP