Логотип Emacs | |
Парадигма | Функциональный, мета, отражающий |
---|---|
Семейство | Лисп |
Разработано | Ричардом Столлманом,. Гаем Л. Стилом-младшим |
Разработчик | Проект GNU |
Впервые появилось | 1985; 35 лет назад (1985 г.) |
Стабильный выпуск | 26.2 / 12 апреля 2019 г.; 18 месяцев назад (12.04.2019) |
Дисциплина печати | Динамический, сильный |
Область действия | Динамический, необязательно лексический |
Платформа | Emacs |
OS | Cross -platform |
Лицензия | GPLv3 |
Расширения имен файлов | .el,.elc |
Под влиянием | |
Common Lisp, Maclisp |
Emacs Lisp диалект языка программирования Lisp, используемый в качестве языка сценариев в Emacs (семейство, наиболее часто связанное с GNU Emacs и XEmacs ). Он используется для реализации большинства функций редактирования, встроенных в Emacs, остальная часть написана на C, как и интерпретатор Лиспа . Emacs Lisp также называется Elisp, хотя существует также более старый, не связанный с ним диалект Lisp с этим именем.
Пользователи Emacs обычно пишут код Emacs Lisp для настройки и расширения Emacs. Другие опции включают функцию Customize, которая присутствует в GNU Emacs с версии 20. Сама по себе, написанная на Emacs Lisp, Customize предоставляет набор страниц настроек, позволяющих пользователю устанавливать опции и предварительно просматривать их эффект в запущенном сеансе Emacs. Когда пользователь сохраняет свои изменения, Customize просто записывает необходимый код Emacs Lisp в файл конфигурации пользователя, который может быть установлен в специальный файл, который использует только Customize, чтобы избежать возможности изменения собственного файла пользователя..
Emacs Lisp также может функционировать как язык сценариев, во многом как Unix оболочка Bourne или Perl, вызов Emacs в пакетном режиме. Таким образом, он может быть вызван из командной строки или через исполняемый файл, а его функции редактирования, такие как буферы и команды перемещения, доступны программе так же, как в обычном режиме. Нет пользовательского интерфейса, когда Emacs запущен в пакетном режиме; он просто выполняет переданный сценарий и завершает работу, отображая любой вывод сценария.
Emacs Lisp наиболее тесно связан с Maclisp, с некоторым более поздним влиянием Common Lisp. Он поддерживает методы императивного и функционального программирования. Ричард Столлман выбрал Lisp в качестве языка расширения для своего переписывания Emacs (в оригинале использовался Text Editor and Corrector (TECO) в качестве языка расширения) из-за его мощных функций, включая возможность рассматривать функции как данные. Хотя стандарт Common Lisp еще не был сформулирован, Scheme существовала в то время, когда Столлман переписывал Gosling Emacs в GNU Emacs. Он решил не использовать его из-за его сравнительно низкой производительности на рабочих станциях (в отличие от мини-компьютеров, которые были традиционным домом Emacs), и он хотел разработать диалект, который, по его мнению, было бы легче оптимизировать.
Диалект Лиспа, используемый в Emacs, существенно отличается от более современных диалектов Common Lisp и Scheme, используемых для программирования приложений. Характерной особенностью Emacs Lisp является использование динамической, а не лексической области видимости по умолчанию (см. Ниже). То есть функция может ссылаться на локальные переменные в той области видимости, из которой она вызывается, но не в той области, где она была определена.
Чтобы понять логику Emacs Lisp, важно помнить, что упор делается на предоставление структур данных и функций, специфичных для создания универсального текстового редактора, а не на реализацию универсального текстового редактора. язык программирования. Например, Emacs Lisp не может легко прочитать файл по строке за раз - весь файл должен быть прочитан в буфер Emacs. Однако Emacs Lisp предоставляет множество функций для навигации и изменения текста в буфере предложения, абзаца или более высокого синтаксического уровня, как определено режимами.
Здесь следует простой пример расширения Emacs, написанного на Emacs Lisp. В Emacs область редактирования может быть разделена на отдельные области, называемые окнами, каждая из которых отображает отдельный буфер. Буфер - это область текста, загруженная в память Emacs (возможно, из файла), которую можно сохранить в текстовый документ.
Пользователи могут нажать стандартную C-x 2привязку клавиш, чтобы открыть новое окно. Это запускает функцию Emacs Lisp split-window-below
. Обычно, когда появляется новое окно, оно отображает тот же буфер, что и предыдущее. Предположим, мы хотим, чтобы он отображал следующий доступный буфер. Для этого пользователь пишет следующий код Emacs Lisp либо в существующий исходный файл Emacs Lisp, либо в пустой буфер Emacs:
(defun my-split-window-func () (interactive) (split-window -below) (set-window-buffer (next-window) (other-buffer))) (global-set-key "\ C-x2" 'my-split-window-func)
Первый оператор, (defun...)
, определяет новую функцию, my-split-window-func
, которая вызывает split-window-below
(старая функция разделения окна function), затем сообщает новому окну отобразить другой (новый) буфер. Второй оператор, (global-set-key...)
повторно связывает последовательность клавиш «C-x 2» с новой функцией.
Это также можно записать с помощью функции, называемой advice, которая позволяет пользователю создавать оболочки вокруг существующих функций вместо определения своих собственных. Это имеет то преимущество, что не требует изменения сочетаний клавиш и работает везде, где вызывается исходная функция, а также его проще писать, но недостатком является усложнение отладки. По этой причине совет не разрешен в исходном коде GNU Emacs, но, если пользователь желает, функция совета может быть использована в его коде для повторной реализации приведенного выше кода следующим образом:
(defadvice split-window-below ( after my-window-splitting-advice first () activate) (set-window-buffer (next-window) (other-buffer)))
Это дает команду split-window-below
выполнить пользовательский код всякий раз, когда он вызывается, перед выполнением остальной части функции. Совет также можно указать для выполнения после исходной функции, вокруг нее - буквально обертывая исходную функцию, или для условного выполнения исходной функции на основе результатов рекомендации.
Emacs 24.4 заменяет этот механизм defadvice
на advice-add
, который считается более гибким и простым. Приведенный выше совет может быть повторно реализован с использованием новой системы как:
(defun switch-to-next-window-in-split () (set-window-buffer (next-window) (other-buffer))) (advice -add 'split-window-below: before #' switch-to-next-window-in-split)
Эти изменения вступают в силу, как только код оценивается. Нет необходимости перекомпилировать, перезапускать Emacs или даже повторно хешировать файл конфигурации. Если код сохранен в файле инициализации Emacs, то Emacs загрузит расширение при следующем запуске. В противном случае при перезапуске Emacs изменения должны быть повторно оценены вручную.
Код Emacs Lisp хранится в файловых системах в виде файлов с обычным текстом, по соглашению с суффиксом имени файла «.el". Файл инициализации пользователя - исключение, часто появляющееся как ".emacs", несмотря на то, что он оценивается как любой код Emacs Lisp. Последние версии Emacs («последние» в программе 40-летней давности означают примерно любую версию, выпущенную с середины 1990-х годов) также будут загружать ~ /.emacs.elи ~ /.emacs.d / init.el. Кроме того, пользователи могут указать любой файл для загрузки в качестве файла конфигурации в командной строке или явно указать, что никакой файл конфигурации не должен загружаться. Когда файлы загружены, компонент интерпретатора программы Emacs читает и анализирует функции и переменные, сохраняя их в памяти. Затем они становятся доступными для других функций редактирования и пользовательских команд. Функции и переменные можно свободно изменять и переопределять без перезапуска редактора или перезагрузки файла конфигурации.
В целях экономии времени и памяти большая часть функциональных возможностей Emacs загружается только при необходимости. Каждый набор дополнительных функций, поставляемых с Emacs, реализуется набором кода Emacs, который называется пакет или библиотека. Например, есть библиотека для выделения ключевых слов в исходном коде программы и библиотека для игры в Тетрис. Каждая библиотека реализована с использованием одного или нескольких исходных файлов Emacs Lisp. Библиотеки могут определять один или несколько основных режимов для активации и управления их функциями.
Разработчики Emacs пишут определенные функции на C. Это примитивы, также называемые встроенными функциями или подгруппами. Хотя примитивы могут быть вызваны из кода Lisp, их можно изменить только путем редактирования исходных файлов C и перекомпиляции. В GNU Emacs примитивы недоступны как внешние библиотеки; они являются частью исполняемого файла Emacs. В XEmacs загрузка таких примитивов во время выполнения возможна с использованием поддержки операционной системой динамического связывания. Функции могут быть написаны как примитивы, потому что им нужен доступ к внешним данным и библиотекам, которые иначе недоступны в Emacs Lisp, или потому, что они вызываются достаточно часто, так что сравнительная скорость C и Emacs Lisp имеет важное значение.
Однако, поскольку ошибки в коде C могут легко привести к нарушениям сегментации или более тонким ошибкам, которые приводят к сбою редактора, и потому что написание кода C, который правильно взаимодействует с Emacs Lisp сборщик мусора подвержен ошибкам, количество функций, реализованных как примитивы, сведено к необходимому минимуму.
Байт-компиляция может ускорить выполнение кода Emacs Lisp. Emacs содержит компилятор, который может переводить исходные файлы Emacs Lisp в специальное представление, называемое байт-кодом . Файлы байт-кода Emacs Lisp имеют суффикс имени файла ".elc". По сравнению с исходными файлами файлы байт-кода загружаются быстрее, занимают меньше места на диске, используют меньше памяти при загрузке и работают быстрее.
Байт-код по-прежнему работает медленнее, чем примитивы, но функции, загружаемые как байт-код, можно легко изменить и повторно загрузить. Кроме того, файлы байт-кода не зависят от платформы. Стандартный код Emacs Lisp, распространяемый вместе с Emacs, загружается как байт-код, хотя соответствующие исходные файлы также обычно предоставляются для справки пользователя. Поставляемые пользователем расширения обычно не компилируются побайтово, поскольку они не такие большие и не так требовательны к вычислительным ресурсам.
Примечательно, что пакет "cl" реализует довольно большое подмножество Common Lisp.
Emacs Lisp (в отличие от некоторых других реализаций Lisp) не поддерживает оптимизация хвостового вызова. Без этого хвостовые рекурсии могут в конечном итоге привести к переполнению стека..
Библиотека помогает писать переносимый код Emacs Lisp с помощью моста платформы polysylabi.
Emacs Lisp - это Lisp-2, что означает, что у него есть пространство имен функций, которое отделено от пространства имен, которое он использует для других переменных.
Как и MacLisp, Emacs Lisp использует динамическую область видимости, предлагая статическую (или лексическую) в качестве опции, начиная с версии 24. Ее можно активировать, установив локальную переменную файла lexical-binding.
.
В динамической области видимости, если программист объявляет переменную в рамках функции, она доступна для подпрограмм, вызываемых из этой функции. Первоначально это было задумано как оптимизация ; лексическая область видимости все еще была редкостью и неопределенной эффективностью. «Я спросил RMS, когда он реализовывал emacs lisp, почему у него динамическая область видимости, и он точно ответил, что лексическая область видимости слишком неэффективна». Динамическое определение также предназначено для обеспечения большей гибкости для пользовательских настроек. Однако динамическое определение области имеет несколько недостатков. Во-первых, это может легко привести к ошибкам в больших программах из-за непреднамеренного взаимодействия между переменными в различных функциях. Во-вторых, доступ к переменным при динамической области видимости обычно медленнее, чем при лексической области видимости.
Кроме того, макрос lexical-let
в пакете «cl» действительно обеспечивает эффективную лексическую область видимости для программистов Emacs Lisp, но хотя "cl" широко используется, lexical-let
используется редко.
Викискладе есть носители, относящиеся к Emacs Lisp. |