Хвостовой вызов

редактировать

В информатике хвостовой вызов представляет собой подпрограмму вызов выполняется как финальное действие процедуры. Если целью хвоста является та же самая подпрограмма, подпрограмма называется хвостовой рекурсией, что является частным случаем прямой рекурсии. Хвостовая рекурсия (или хвостовая рекурсия ) особенно полезна, и ее часто легко обрабатывать в реализациях.

Хвостовые вызовы могут быть реализованы без добавления нового кадра стека в стек вызовов . Большая часть кадра текущей процедуры больше не нужна и может быть заменена кадром хвостового вызова, измененным соответствующим образом (аналогично overlay для процессов, но для вызовов функций). Затем программа может перейти к вызываемой подпрограмме. Создание такого кода вместо стандартной последовательности вызовов называется устранением хвостового вызова или оптимизацией хвостового вызова . Устранение хвостового вызова позволяет реализовать вызовы процедур в хвостовой позиции так же эффективно, как и операторы goto, что обеспечивает эффективное структурное программирование. По словам Гая Л. Стила, «в общем, вызовы процедур можно с пользой рассматривать как операторы GOTO, которые также передают параметры и могут быть единообразно закодированы как инструкции JUMP [машинный код]».

Не все языки программирования требуют исключения хвостового вызова. Однако в языках функционального программирования устранение хвостового вызова часто гарантируется языковым стандартом , что позволяет хвостовой рекурсии использовать такой же объем памяти, как и эквивалентный цикл . Особый случай хвостовых рекурсивных вызовов, когда функция вызывает саму себя, может быть более податливым для исключения вызова, чем общие хвостовые вызовы. Когда семантика языка явно не поддерживает общие хвостовые вызовы, компилятор часто все еще может оптимизировать родственные вызовы или хвостовые вызовы функций, которые принимают и возвращают те же типы, что и вызывающий.

Содержание
  • 1 Описание
  • 2 Синтаксическая форма
  • 3 Примеры программ
  • 4 Хвостовая рекурсия по модулю cons
    • 4.1 Пример пролога
    • 4.2 Пример C
  • 5 История
  • 6 Методы реализации
    • 6.1 In сборка
    • 6.2 Через трамплинги
  • 7 Связь с конструкцией while
  • 8 По языку
  • 9 См. также
  • 10 Примечания
  • 11 Ссылки
Описание

Когда функция При вызове компьютер должен «запомнить» место, откуда он был вызван, адрес возврата, чтобы он мог вернуться в это место с результатом после завершения вызова. Как правило, эта информация сохраняется в стеке вызовов , простом списке местоположений возврата в порядке времени достижения описываемых ими местоположений вызовов. Для хвостовых вызовов нет необходимости запоминать вызывающего - вместо этого удаление хвостового вызова вносит только минимально необходимые изменения в кадр стека перед его передачей, а функция, вызываемая хвостом, вернется непосредственно к исходному вызывающему. Хвостовой вызов не обязательно должен появляться лексически после всех других операторов в исходном коде; Важно только, чтобы вызывающая функция возвращалась сразу после хвостового вызова, возвращая результат хвостового вызова, если таковой имеется, поскольку вызывающая функция игнорируется при выполнении оптимизации.

Для нерекурсивных вызовов функций это обычно оптимизация, которая экономит лишь немного времени и места, так как не так много различных функций, доступных для вызова. Однако при работе с рекурсивными или взаимно рекурсивными функциями, где рекурсия происходит через хвостовые вызовы, пространство стека и количество сохраненных возвратов могут вырасти и стать очень значительными, поскольку функция может вызывать сама себя, прямо или косвенно, каждый раз создавая новый кадр стека вызовов. Устранение хвостового вызова часто снижает требования к асимптотическому пространству стека с линейных, или O (n), до постоянных, или O (1). Таким образом, исключение хвостового вызова требуется стандартными определениями некоторых языков программирования, таких как Scheme, и языков семейства ML среди других. Определение языка схемы точно формализует интуитивное понятие положения хвоста, указывая, какие синтаксические формы позволяют получить результаты в контексте хвоста. Реализации, позволяющие неограниченному количеству хвостовых вызовов быть активными в один и тот же момент, благодаря исключению хвостовых вызовов, также могут называться «правильно хвостовыми рекурсивными».

Помимо пространства и эффективности выполнения, важно исключение хвостовых вызовов в идиоме функционального программирования, известной как стиль передачи продолжения (CPS), который в противном случае быстро исчерпал бы пространство стека.

Синтаксическая форма

Хвостовой вызов может располагаться непосредственно перед синтаксическим концом функции:

function foo (data) {a (data); вернуть b (данные); }

Здесь оба a (data)и b (data)являются вызовами, но b- это последнее, что процедура выполняет перед возвратом и является таким образом в хвостовой позиции. Однако не все хвостовые вызовы обязательно расположены в синтаксическом конце подпрограммы:

function bar (data) {if (a (data)) {return b (data); } return c (данные); }

Здесь оба вызова bи cнаходятся в хвостовой позиции. Это связано с тем, что каждый из них находится в конце if-ветки соответственно, даже если первый синтаксически не находится в конце тела bar.

В этом коде:

function foo1 (data) {return a (data) + 1; }
функция foo2 (данные) {var ret = a (данные); return ret; }
функция foo3 (данные) {var ret = a (данные); возврат (ret == 0)? 1: рет; }

вызов a (data)находится в хвостовой позиции в foo2, но это не в хвостовой позиции либо в foo1или в foo3, потому что управление должно возвращаться вызывающему, чтобы он мог проверить или изменить возвращаемое значение перед его возвратом.

Примеры программ

Следующая программа является примером на Схеме :

;; факториал: число ->число ;; вычислить произведение всех положительных ;; целые числа, меньшие или равные n. (define (factorial n) (if (= n 1) 1 (* n (factorial (- n 1)))))

Это не написано в стиле хвостовой рекурсии, потому что функция умножения ("*") находится в положении хвоста. Это можно сравнить с:

;; факториал: число ->число ;; вычислить произведение всех положительных ;; целые числа, меньшие или равные n. (define (factorial n) (fact-iter 1 n)) (define (fact-iter product n) (if (< n 2) product (fact-iter (* product n) (- n 1))))

Эта программа предполагает оценку аппликативного порядка. Внутренняя процедура fact -iterвызывает себя последним в потоке управления. Это позволяет интерпретатору или компилятору реорганизовать выполнение, которое обычно выглядит следующим образом:

call factorial (4) call fact-iter (1 4) call fact-iter (4 3) call fact-iter (12 2) call fact-iter (24 1) return 24 return 24 return 24 return 24 return 24

в более эффективный вариант с точки зрения пространства и времени:

вызов факториала (4) вызов факториала (1 4) замените аргументы на (4 3) замените аргументы на (12 2) заменить аргументы на (24 1) return 24 return 24

Эта реорганизация экономит место, поскольку не требуется сохранять состояние, за исключением адреса вызывающей функции, ни в стеке, ни в куче, а кадр стека вызовов для fact-iterповторно используется для хранения промежуточных результатов. lso означает, что программисту не нужно беспокоиться о нехватке стека или кучи для очень глубоких рекурсий. В типичных реализациях хвостовой рекурсивный вариант будет значительно быстрее, чем другой вариант, но только с постоянным коэффициентом.

Некоторые программисты, работающие с функциональными языками, перепишут рекурсивный код, чтобы он стал хвостовым, чтобы они могли воспользоваться этой функцией. Это часто требует добавления в функцию аргумента «аккумулятор» (productв приведенном выше примере). В некоторых случаях (например, в списках фильтрации) и на некоторых языках полная хвостовая рекурсия может потребовать написания функции, которая ранее была чисто функциональной, чтобы она изменяла ссылки, хранящиеся в других переменных.

Хвостовая рекурсия по модулю cons

Хвостовая рекурсия по модулю cons - это обобщение оптимизации хвостовой рекурсии, представленное Дэвидом HD Уорреном в контексте компиляции из Пролога, рассматриваемой как явно установить один раз язык. Он был описан (хотя и не назван) Дэниелом П. Фридманом и в 1974 году как метод компиляции LISP. Как следует из названия, он применяется, когда единственная операция, которую нужно выполнить после рекурсивного вызова, - это добавить известное значение перед списком, возвращаемым из него (или, как правило, выполнить постоянное количество простых операций по построению данных). Таким образом, этот вызов был бы хвостовым вызовом, за исключением ("по модулю ") указанной операции cons. Но префикс значения в начале списка при выходе из рекурсивного вызова - это то же самое, что добавление этого значения в конец растущего списка при входе в рекурсивный вызов, тем самым создавая список как побочный эффект , как будто в неявном параметре аккумулятора. Следующий фрагмент Пролога иллюстрирует эту концепцию:

Пример Пролога

% хвостовой рекурсии по модулю cons: partition (, _,). partition ([X | Xs], Pivot, [X | Rest], Bigs): - X @ < Pivot, !, partition(Xs, Pivot, Rest, Bigs). partition([X|Xs], Pivot, Smalls, [X|Rest]) :- partition(Xs, Pivot, Smalls, Rest).
- В Haskell защищенная рекурсия: partition :: Ord a =>[a] ->a ->([ a], [a]) раздел _ = (,) раздел (x: xs) p | x < p = (x:a,b) | otherwise = (a,x:b) where (a,b) = partition xs p
% С явными унификациями:% не хвостовая рекурсивная трансляция: раздел (, _,). раздел (L, Pivot, Smalls, Bigs): - L = [X | Xs], (X @ < Pivot ->partition (Xs, Pivot, Rest, Bigs), Smalls = [X | Rest]; раздел (Xs, Pivot), Small, Rest), Bigs = [X | Rest]).
% С явными унификациями:% хвостовой рекурсивный перевод: раздел (, _,). раздел (L, Pivot, Smalls, Bigs): - L = [X | Xs], (X @ < Pivot ->Smalls = [X | Rest], раздел (Xs, Pivot, Rest, Bigs); Bigs = [X | Rest], перегородка (Xs, Pivot, Smalls, Rest)).

Таким образом, при хвостовой рекурсивной трансляции такой вызов преобразуется в сначала создание нового узла списка и установку его первого поля, а затем выполнение хвостового вызова с указателем на узел Поле restв качестве аргумента, которое должно быть заполнено рекурсивно.

Пример C

Следующий фрагмент определяет рекурсивную функцию в C, которая дублирует связанный список:

typedef struct list {void * value; список структур * следующий; } список; список * дубликат (константный список * ls) {список * голова = NULL; if (ls! = NULL) {список * p = дубликат (ls->следующий); голова = malloc (размер * голова); head->value = ls->value; head->next = p; } вернуть голову; }
;; в схеме (define (duplicate ls) (if (not (null? ls)) (cons (car ls) (duplicate (cdr ls))) '()))
%% в Prolog, dup ([X | Xs], R): - dup (Xs, Ys), R = [X | Ys]. dup (,).

В этой форме функция не является хвостовой рекурсивной, потому что управление возвращается вызывающей стороне после того, как рекурсивный вызов дублирует остальную часть входного списка. Даже если бы он выделил головной узел перед дублированием остальных, ему все равно нужно было бы вставить результат рекурсивного вызова в поле nextпосле вызова. Таким образом, функция почти хвостовая рекурсивная. Метод Уоррена накладывает ответственность за заполнение поля nextна сам рекурсивный вызов, который, таким образом, становится хвостовым вызовом:

typedef struct list {void * value; список структур * следующий; } список; void duplicate_aux (список констант * ls, список * конец); список * дубликат (const list * ls) {заголовок списка; duplicate_aux (ls, head); return head.next; } void duplicate_aux (const list * ls, list * end) {if (ls! = NULL) {end->next = malloc (sizeof * end); конец->следующий->значение = ls->значение; duplicate_aux (ls->следующий, конец->следующий); } else {конец->следующий = NULL; }}
;; в схеме (define (duplicate ls) (let ((head (list 1))) (let dup ((ls ls) (end head)) (cond ((not (null? ls)) (set-cdr! end (list (car ls))) (dup (cdr ls) (cdr end))))) (cdr head)))
%% в Prolog, dup ([X | Xs], R): - R = [X | Ys], дубликат (Xs, Ys). dup (,).

(Контрольный головной узел используется для упрощения кода.) Теперь вызываемый абонент присоединяется к концу растущего списка, а не заставляет вызывающего абонента добавлять его в начало возвращенного списка. Теперь работа выполняется на пути вперед от начала списка до рекурсивного вызова, который затем продолжается, а не назад от конца списка, после того, как рекурсивный вызов вернул свой результат. Таким образом, он похож на метод накопления параметров, превращая рекурсивное вычисление в итерационное.

Характерно для этого метода, в стеке вызовов выполнения создается родительский кадр , который хвостовой рекурсивный вызываемый объект может повторно использовать в качестве своего собственного кадра вызова, если присутствует оптимизация хвостового вызова.

Реализация хвостовой рекурсии теперь может быть преобразована в явно итеративную форму, как накопительный цикл :

typedef struct list {void * value; список структур * следующий; } список; list * duplicate (const list * ls) {заголовок списка, * конец; конец = голова; в то время как (ls! = NULL) {конец->следующий = malloc (sizeof * end); конец->следующий->значение = ls->значение; ls = ls->следующий; конец = конец->следующий; } конец->следующий = NULL; return head.next; }
;; в схеме (define (duplicate ls) (let ((head (list 1))) (do ((end head (cdr end)) (ls ls (cdr ls))) ((null? ls) (cdr head)) (set-cdr! end (list (car ls))))))
%% в Прологе, %% Н / Д
История

В документе, доставленном в На конференции ACM в Сиэтле в 1977 году Гай Л. Стил подвел итоги дебатов по GOTO и структурированному программированию и заметил, что вызовы процедур в хвосте Положение процедуры лучше всего рассматривать как прямую передачу управления вызываемой процедуре, обычно устраняя ненужные операции манипулирования стеком. Поскольку такие «хвостовые вызовы» очень распространены в Lisp, языке, где вызовы процедур являются повсеместными, эта форма оптимизации значительно снижает стоимость вызова процедуры по сравнению с другими реализациями. Стил утверждал, что плохо реализованные вызовы процедур привели к искусственному восприятию того, что GOTO дешевле, чем вызов процедуры. Стил далее утверждал, что «в целом вызовы процедур можно с пользой рассматривать как операторы GOTO, которые также передают параметры и могут быть единообразно закодированы как инструкции JUMP [машинного кода]», при этом инструкции манипулирования стеком машинного кода «считаются оптимизацией (а не наоборот!)". Стил привел доказательства того, что хорошо оптимизированные числовые алгоритмы в Лиспе могут выполняться быстрее, чем код, созданный доступными в то время коммерческими компиляторами Фортрана, потому что стоимость вызова процедуры в Лиспе была намного ниже. В Scheme, диалекте Лиспа, разработанном Стилом с Джеральдом Джей Сассманом, исключение хвостового вызова гарантированно реализовано в любом интерпретаторе.

Методы реализации

Хвостовая рекурсия важна для некоторых языков высокого уровня, особенно языков функциональной и логики и членов семейства Lisp. В этих языках хвостовая рекурсия является наиболее часто используемым (а иногда и единственным доступным) способом реализации итерации. Спецификация языка Scheme требует, чтобы хвостовые вызовы были оптимизированы, чтобы не увеличивать стек. Хвостовые вызовы могут быть сделаны явно в Perl, с вариантом оператора "goto", который принимает имя функции: goto NAME;

Однако для языковых реализаций, которые хранят аргументы функции и локальные переменные в стеке вызовов (который является реализацией по умолчанию для многих языков, по крайней мере, в системах с аппаратным стеком , таких как x86 ), реализующие обобщенные Оптимизация хвостового вызова (включая взаимную хвостовую рекурсию) представляет проблему: если размер записи активации вызываемого абонента отличается от размера вызывающего, то может потребоваться дополнительная очистка или изменение размера кадра стека. В этих случаях оптимизация хвостовой рекурсии остается тривиальной, но общую оптимизацию хвостовых вызовов может быть труднее реализовать эффективно.

Например, в виртуальной машине Java (JVM) хвостовые рекурсивные вызовы могут быть исключены (поскольку при этом повторно используется существующий стек вызовов), но общие хвостовые вызовы не могут быть (как это изменяет стек вызовов). В результате функциональные языки, такие как Scala, ориентированные на JVM, могут эффективно реализовывать прямую хвостовую рекурсию, но не взаимную хвостовую рекурсию.

Компиляторы GCC, LLVM / Clang и Intel выполняют оптимизацию хвостового вызова для C и других языков на более высоких уровнях оптимизации или когда передана опция -foptimize-sibling-calls. Хотя данный синтаксис языка может не поддерживать его явно, компилятор может выполнить эту оптимизацию всякий раз, когда он может определить, что типы возвращаемых значений для вызывающего и вызываемого эквивалентны и что типы аргументов, переданные в обе функции, либо одинаковы, либо требуют такой же общий объем памяти в стеке вызовов.

Доступны различные методы реализации.

В сборке

Хвостовые вызовы часто оптимизируются интерпретаторами и компиляторами функционального программирования и логического программирования языков к более эффективным формам итерации. Например, программисты схемы обычно выражают циклы while как вызовы процедур в хвостовой позиции и полагаются на компилятор или интерпретатор схемы, чтобы заменить хвостовые вызовы более эффективными jump инструкции.

Для компиляторов, генерирующих сборку напрямую, исключение хвостового вызова выполняется просто: достаточно заменить код операции вызова на код перехода после фиксации параметров в стеке. С точки зрения компилятора, первый приведенный выше пример изначально переведен на псевдо язык ассемблера (на самом деле это действительно x86 assembly ):

foo: call B call A ret

Устранение хвостового вызова заменяет последние две строки одной инструкцией перехода:

foo: call B jmp A

После завершения подпрограммы Aон вернется непосредственно на адрес возврата foo, опуская ненужный оператор ret.

Обычно вызываемые подпрограммы должны быть снабжены параметрами. Таким образом, сгенерированный код должен убедиться, что кадр вызова для A правильно настроен, прежде чем переходить к подпрограмме, вызываемой хвостом. Например, на платформах, где стек вызовов не только содержит адрес возврата, но также и параметры для подпрограммы, компилятору может потребоваться передать инструкции для настройки стека вызовов. На такой платформе для кода:

function foo (data1, data2) B (data1) return A (data2)

(где data1и data2- параметры) компилятор может преобразовать это как:

1 foo: 2 mov reg, [sp + data1]; получить data1 из параметра стека (sp) в рабочий регистр. 3 push reg; поместить data1 в стек, где B ожидает 4 call B; B использует данные1 5 pop; удалить data1 из стека 6 mov reg, [sp + data2]; получить данные2 из параметра стека (sp) в рабочий регистр. 7 push reg; поместить data2 в стек, где A ожидает 8 call A; A использует data2 9 pop; удалить data2 из стека. 10 ret

Оптимизатор хвостового вызова может затем изменить код на:

1 foo: 2 mov reg, [sp + data1]; получить data1 из параметра стека (sp) в рабочий регистр. 3 нажатия рег; поместить data1 в стек, где B ожидает 4 call B; B использует данные1 5 pop; удалить data1 из стека 6 mov reg, [sp + data2]; получить данные2 из параметра стека (sp) в рабочий регистр. 7 mov [sp + data1], reg; поместите data2 туда, где A ожидает 8 jmp A; A использует data2 и немедленно возвращается вызывающему.

Этот код более эффективен как с точки зрения скорости выполнения, так и с точки зрения использования пространства стека.

Через trampolining

Поскольку многие компиляторы Scheme используют C в качестве промежуточного целевого кода, хвостовая рекурсия должна быть закодирована на C без увеличения стека, даже если компилятор C не оптимизирует хвостовые вызовы. Многие реализации достигают этого с помощью устройства, известного как батут, фрагмента кода, который многократно вызывает функции. Все функции вводятся через батут. Когда функция должна следовать за другой функцией, вместо того, чтобы вызывать ее напрямую и затем возвращать результат, она возвращает адрес вызываемой функции и параметры вызова обратно в трамплин (с которого он был вызван сам), а trampoline позаботится о следующем вызове этой функции с указанными параметрами. Это гарантирует, что стек C не будет расти, и итерация может продолжаться бесконечно.

Можно реализовать батуты с помощью функций высшего порядка на языках, которые их поддерживают, таких как Groovy, Visual Basic.NET и C#.

Использование трамплина для всех вызовов функций намного дороже, чем обычный вызов функции C, поэтому по крайней мере один компилятор схемы, Chicken, использует метод, впервые описанный Генри Бейкером из неопубликованного предложения Эндрю Аппеля, в котором используются обычные вызовы C, но размер стека проверяется перед каждым вызовом. Когда стек достигает максимально допустимого размера, объекты в стеке собираются сборщиком мусора с использованием алгоритма Чейни путем перемещения всех оперативных данных в отдельную кучу. После этого стек разматывается («выталкивается»), и программа возобновляет работу из состояния, сохраненного непосредственно перед сборкой мусора. Бейкер говорит: «Метод Аппеля позволяет избежать большого количества прыжков на батуте, иногда прыгая с Эмпайр-стейт-билдинг». Сборка мусора гарантирует, что взаимная хвостовая рекурсия может продолжаться бесконечно. Однако этот подход требует, чтобы ни один вызов функции C никогда не возвращался, поскольку нет гарантии, что фрейм стека вызывающей стороны все еще существует; следовательно, он включает в себя гораздо более драматическое внутреннее переписывание программного кода: стиль передачи продолжения.

Отношение к конструкции while

Хвостовая рекурсия может быть связана с while оператор потока управления посредством преобразования, например следующего:

функция foo (x) is: ifpredicate (x) затемreturn foo (bar (x)) elsereturn baz (x)

Приведенная выше конструкция преобразуется в:

function foo (x) is: while predicate (x) do : x ← bar (x) return baz (x)

Ранее x может быть кортежем, включающим более одной переменной: если это так, то при разработке оператора присваивания x ← bar (x) необходимо проявлять осторожность, чтобы соблюдались зависимости. Может потребоваться ввести вспомогательные переменные или использовать конструкцию swap.

Более общее использование хвостовой рекурсии может быть связано с операторами потока управления, такими как break и continue, как показано ниже:

function foo (x) is: ifp (x), затемвернуть bar (x) иначе, если q (x), затемreturn baz (x)... else if t (x) затемreturn foo (quux (x))... else return foo (quuux (x))

где bar и baz - прямые обратные вызовы, тогда как quux и quuux включают рекурсивный хвостовой вызов foo. Перевод дается следующим образом:

function foo (x) is: do: ifp (x) then x ← bar (x) breakelse if q (x) затем x ← baz (x) break ... else if t (x) то x ← quux (x) continue ... else x ← quuux (x) continueloopreturn x
Автор language
  • Haskell - Да
  • Erlang - Да
  • Common Lisp - Некоторые реализации выполняют оптимизацию хвостового вызова во время компиляции при оптимизации для скорости
  • JavaScript - ECMAScript 6.0 совместимые движки должны иметь хвостовые вызовы, которые теперь реализованы в Safari / WebKit, но отклонены V8 и SpiderMonkey
  • Lua - хвостовая рекурсия требуется определением языка
  • Python - Стандартные реализации Python не выполняют оптимизацию хвостового вызова, хотя для этого доступен сторонний модуль. Изобретатель языка Гвидо ван Россум утверждает, что трассировки стека изменяются путем исключения хвостового вызова, что усложняет отладку, и предпочитает, чтобы программисты использовали явную итерацию вместо
  • Rust - оптимизация хвостового вызова может выполняться в ограниченных случаях, но не гарантируется
  • Схема - Требуется определением языка
  • Racket - Да
  • Tcl - Поскольку Tcl 8.6, Tcl имеет команду tailcall
  • Kotlin - имеет модификатор tailrecдля функций
  • Elixir - Elixir реализует оптимизацию хвостового вызова Как и все языки, нацеленные в настоящее время на виртуальную машину BEAM.
  • Perl - явный вариант с вариантом оператора "goto", который принимает имя функции: goto NAME;
  • Scala - хвостовые рекурсивные функции автоматически оптимизируются компилятором. Такие функции также могут быть помечены аннотацией @tailrec, что делает ошибку компиляции, если функция не является хвостовой рекурсивной
  • Objective-C - Компилятор оптимизирует хвостовые вызовы, когда -O1 ( или выше), но ее легко нарушить вызовы, добавленные с помощью Automatic Reference Counting (ARC).
  • F# - F # реализует TCO по умолчанию, где это возможно
  • Clojure - Clojure имеет recurспециальная форма.
См. также
  • значок Портал компьютерного программирования
Найдите хвостовую рекурсию в Wiktionary, бесплатном словаре.
Примечания
Ссылки

Эта статья основана на взятом материале из Бесплатный он-лайн словарь по вычислительной технике до 1 ноября 2008 г. и включенный в соответствии с условиями «перелицензирования» GFDL версии 1.3 или более поздней.

Последняя правка сделана 2021-06-09 07:43:36
Содержание доступно по лицензии CC BY-SA 3.0 (если не указано иное).
Обратная связь: support@alphapedia.ru
Соглашение
О проекте