Макросъемки (сокращенно «макрокоманда», от греческого сочетания формы μακρο- «длинный, большой») в информатике является правилом или шаблон, который определяет, как некий вход должен быть сопоставлен с выходом замены. Применение макроса к входу - это расширение макроса. Входные и выходные данные могут быть последовательностью лексических знаков или символов или синтаксическим деревом. Символьные макросы поддерживаются в программных приложениях, чтобы упростить вызов общих последовательностей команд. В некоторых языках программирования поддерживаются макросы токена и дерева, позволяющие повторно использовать код или расширять язык, иногда для языков, специфичных для предметной области.
Макросы используются, чтобы сделать последовательность вычислительных инструкций доступной для программиста в виде единого программного оператора, что делает задачу программирования менее утомительной и менее подверженной ошибкам. (Таким образом, они называются «макросами», потому что «большой» блок кода может быть расширен из «маленькой» последовательности символов.) Макросы часто допускают позиционные или ключевые параметры, которые определяют, что генерирует программа условного ассемблера и которые используются для создавать целые программы или программные пакеты в соответствии с такими переменными, как операционная система, платформа или другие факторы. Термин происходит от «макро-инструкции», и такие расширения изначально использовались при генерации кода на языке ассемблера.
Клавиатурные макросы и макросы мыши позволяют короткие последовательности нажатий клавиш и действий мыши превратить в другие, как правило, больше времени, последовательности нажатий клавиш и действий мыши. Таким образом можно автоматизировать часто используемые или повторяющиеся последовательности нажатий клавиш и движений мыши. Отдельные программы для создания этих макросов называются регистраторами макросов.
В течение 1980-х годов макропрограммы - первоначально SmartKey, затем SuperKey, KeyWorks, Prokey - были очень популярны, сначала как средство автоматического форматирования сценариев, а затем для множества задач пользовательского ввода. Эти программы были основаны на режиме работы TSR ( завершить и оставаться резидентным ) и применялись ко всему вводу с клавиатуры, независимо от того, в каком контексте это происходило. Они в некоторой степени устарели после появления пользовательских интерфейсов, управляемых мышью, и появления макросов клавиатуры и мыши в таких приложениях, как текстовые процессоры и электронные таблицы, что сделало возможным создание чувствительных к приложениям макросов клавиатуры.
Макросы клавиатуры можно использовать в многопользовательских ролевых онлайн-играх (MMORPG) для выполнения повторяющихся, но прибыльных задач, тем самым накапливая ресурсы. Поскольку это делается без человеческих усилий, это может исказить экономику игры. По этой причине использование макросов является нарушением TOS или EULA большинства MMORPG, и их администраторы тратят значительные усилия на их подавление.
Макросы клавиатуры и мыши, созданные с использованием встроенных в приложение функций макросов, иногда называют макросами приложения. Они создаются путем однократного выполнения последовательности и разрешения приложению записывать действия. Также может существовать базовый язык макросов, чаще всего язык сценариев, с прямым доступом к функциям приложения.
Текстовый редактор программистов Emacs (сокращение от «редактирование макросов») доводит эту идею до конца. Фактически, большая часть редактора состоит из макросов. Первоначально Emacs был разработан как набор макросов на языке редактирования TECO ; позже он был перенесен на диалекты Лиспа.
Текстовый редактор другого программиста, Vim (потомок vi ), также имеет реализацию макросов клавиатуры. Он может записывать в регистр (макрос) то, что человек набирает на клавиатуре, и его можно воспроизводить или редактировать так же, как макросы VBA для Microsoft Office. В Vim также есть скриптовый язык Vimscript для создания макросов.
Visual Basic для приложений (VBA) - это язык программирования, включенный в Microsoft Office от Office 97 до Office 2019 (хотя он был доступен в некоторых компонентах Office до Office 97). Однако его функция эволюционировала и заменила макроязыки, которые изначально были включены в некоторые из этих приложений.
XEDIT, работающий в компоненте Conversational Monitor System (CMS) виртуальной машины, поддерживает макросы, написанные в EXEC, EXEC2 и REXX, а некоторые команды CMS фактически были оболочками для макросов XEDIT. Редактор Hessling (THE), частичный клон XEDIT, поддерживает макросы Rexx с использованием Regina и Open Object REXX (oorexx). Многие распространенные приложения, в том числе и на ПК, используют Rexx в качестве языка сценариев.
VBA имеет доступ к большинству системных вызовов Microsoft Windows и выполняется при открытии документов. Это позволяет относительно легко писать компьютерные вирусы на VBA, обычно известные как макровирусы. В середине-конце 1990-х годов это стало одним из самых распространенных типов компьютерных вирусов. Однако в конце 1990-х и до настоящего времени Microsoft исправляла и обновляла свои программы. Кроме того, современные антивирусные программы сразу же противодействуют таким атакам.
Параметризованный макрос макрос, который способен вставить данные объектов в его расширение. Это дает макросу некоторую мощность функции.
В качестве простого примера на языке программирования C это типичный макрос, который не является параметризованным макросом:
#define PI 3.14159
Это приводит PI
к тому, что всегда заменяется на 3.14159
везде, где это происходит. С другой стороны, пример параметризованного макроса:
#define pred(x) ((x)-1)
То, что расширяется до этого макроса, зависит от того, какой аргумент x ему передан. Вот несколько возможных расширений:
pred(2) → ((2) -1) pred(y+2) → ((y+2) -1) pred(f(5)) → ((f(5))-1)
Параметризованные макросы являются полезным механизмом на уровне исходного кода для выполнения встроенного расширения, но в таких языках, как C, где они используют простую текстовую замену, они имеют ряд серьезных недостатков по сравнению с другими механизмами для выполнения встроенного расширения, такими как встроенные функции..
С другой стороны, параметризованные макросы, используемые в таких языках, как Lisp, PL / I и Scheme, намного мощнее и способны принимать решения о том, какой код создавать, на основе своих аргументов; таким образом, их можно эффективно использовать для генерации кода во время выполнения.
Такие языки, как C и некоторые языки ассемблера, имеют элементарные макросистемы, реализованные как препроцессоры для компилятора или ассемблера. Макросы препроцессора C работают путем простой текстовой подстановки на уровне токена, а не на уровне символа. Однако макро-средства более сложных ассемблеров, например, IBM High Level Assembler (HLASM), не могут быть реализованы с помощью препроцессора; код для сборки инструкций и данных перемежается с кодом для сборки макросов.
Классическое использование макросов - это компьютерная система набора текста TeX и ее производные, где большая часть функциональных возможностей основана на макросах.
MacroML - это экспериментальная система, которая стремится согласовать статическую типизацию и макросистемы. В Nemerle есть типизированные синтаксические макросы, и один из продуктивных способов представить эти синтаксические макросы - это многоступенчатые вычисления.
Другие примеры:
Некоторые основные приложения были написаны как текстовые макросы, вызываемые другими приложениями, например, XEDIT в CMS.
Некоторые языки, такие как PHP, могут быть встроены в текст произвольного формата или исходный код других языков. Механизм распознавания фрагментов кода (например, заключенных в скобки lt;?php
и ?gt;
) аналогичен текстовому макроязыку, но это гораздо более мощные и полнофункциональные языки.
Макросы на языке PL / I написаны в подмножестве самого PL / I: компилятор выполняет « операторы препроцессора » во время компиляции, и выходные данные этого выполнения составляют часть компилируемого кода. Возможность использовать знакомый процедурный язык, поскольку макроязык дает гораздо больше возможностей, чем у макросов подстановки текста, за счет более крупного и медленного компилятора.
Макросы кадров технологии Frame имеют собственный синтаксис команд, но также могут содержать текст на любом языке. Каждый фрейм является как общим компонентом в иерархии вложенных подсборок, так и процедурой интеграции с его фреймами подсборки (рекурсивный процесс, который разрешает конфликты интеграции в пользу подсборок более высокого уровня). На выходе получаются пользовательские документы, обычно компилируемые исходные модули. Фреймовая технология позволяет избежать увеличения числа похожих, но слегка отличающихся компонентов - проблемы, которая преследовала разработку программного обеспечения с момента изобретения макросов и подпрограмм.
Большинство языков ассемблера имеют менее мощные процедурные макросы, например, позволяющие повторять блок кода N раз для развертывания цикла ; но они имеют совершенно другой синтаксис, чем фактический язык ассемблера.
Макросистемы, такие как препроцессор C, описанный ранее, которые работают на уровне лексических токенов, не могут надежно сохранять лексическую структуру. Вместо этого синтаксические макросистемы работают на уровне абстрактных синтаксических деревьев и сохраняют лексическую структуру исходной программы. Наиболее широко используемые реализации синтаксических макросистем находятся в Lisp- подобных языках. Эти языки особенно подходят для этого стиля макросов из-за их единообразного синтаксиса в скобках (известного как S-выражения ). В частности, унифицированный синтаксис упрощает определение вызовов макросов. Макросы Lisp преобразуют саму структуру программы, и для выражения таких преобразований доступен полный язык. Хотя синтаксические макросы часто встречаются в Lisp-подобных языках, они также доступны в других языках, таких как Prolog, Erlang, Dylan, Scala, Nemerle, Rust, Elixir, Nim, Haxe и Julia. Они также доступны как сторонние расширения для JavaScript, C # и Python.
До того, как в Лиспе появились макросы, в нем были так называемые FEXPR, операторы, подобные функциям, входные данные которых были не значениями, вычисляемыми аргументами, а скорее синтаксическими формами аргументов, и чьи выходные данные были значениями, которые должны использоваться в вычислениях. Другими словами, FEXPR были реализованы на том же уровне, что и EVAL, и предоставляли окно на уровень метаоценки. В целом оказалось, что эту модель сложно эффективно осмыслить.
В 1963 году Тимоти Харт предложил добавить макросы в Lisp 1.5 в AI Memo 57: MACRO Definitions for LISP.
Анафорический макрос - это тип программного макроса, который намеренно фиксирует некоторую форму, предоставленную макросу, на которую может ссылаться анафора (выражение, относящееся к другому). Анафорические макросы впервые появились в книге Пола Грэма «О Лиспе», и их название является отсылкой к лингвистической анафоре - использованию слов вместо предшествующих слов.
В середине восьмидесятых годов в ряде статей было введено понятие гигиенического макрорасширения ( syntax-rules
), системы на основе шаблонов, в которой синтаксические среды определения макроса и использования макроса различны, что позволяет разработчикам макросов и пользователям не беспокоиться о непреднамеренных действиях. захват переменных (см. ссылочную прозрачность ). Гигиенические макросы стандартизированы для Scheme в стандартах R5RS, R6RS и R7RS. Ряд конкурирующих реализаций гигиенических макросов существует такие как syntax-rules
, syntax-case
, явное переименование и синтаксическое закрытие. Оба syntax-rules
и syntax-case
были стандартизированы в стандартах Схемы.
Недавно Racket объединил понятия гигиенических макросов с « башней оценщиков », так что время синтаксического расширения одной макросистемы является обычным временем выполнения другого блока кода, и показал, как применять чередование расширения и синтаксического анализа в не -подобный язык.
Ряд языков, отличных от Scheme, либо реализуют гигиенические макросы, либо реализуют частично гигиенические системы. Примеры включают Scala, Rust, Elixir, Julia, Dylan, Nim и Nemerle.
cond
но отсутствует if
, последний можно определить в терминах первого, используя макросы. Например, в Scheme есть как продолжения, так и гигиенические макросы, которые позволяют программисту разрабатывать собственные абстракции управления, такие как циклы и конструкции раннего выхода, без необходимости встраивать их в язык.let
в приложение функции к набору аргументов.Фелляйзен предполагает, что эти три категории составляют основное законное использование макросов в такой системе. Другие предложили альтернативные варианты использования макросов, такие как анафорические макросы в макросистемах, которые негигиеничны или допускают выборочное антигигиеничное преобразование.
Взаимодействие макросов и других языковых функций было продуктивной областью исследований. Например, компоненты и модули полезны для крупномасштабного программирования, но взаимодействие макросов и этих других конструкций должно быть определено для их совместного использования. Модульные и компонентные системы, которые могут взаимодействовать с макросами, были предложены для Scheme и других языков с макросами. Например, язык Racket расширяет понятие макросистемы до синтаксической башни, где макросы могут быть написаны на языках, включая макросы, с использованием гигиены, чтобы гарантировать, что синтаксические слои различны и позволяя модулям экспортировать макросы в другие модули.
Макросы обычно используются для сопоставления короткой строки (вызов макроса) более длинной последовательности инструкций. Другое, менее распространенное использование макросов - обратное: отображение последовательности инструкций в строке макроса. Это был подход, принятый системой мобильного программирования STAGE2, которая использовала элементарный компилятор макросов (называемый SIMCMP) для сопоставления определенного набора инструкций данного компьютера с аналогичными машинно-независимыми макросами. Приложения (особенно компиляторы), написанные с помощью этих машинно-независимых макросов, можно будет запускать без изменений на любом компьютере, оснащенном элементарным компилятором макросов. Первое приложение, запускаемое в таком контексте, - это более сложный и мощный компилятор макросов, написанный на машинно-независимом языке макросов. Этот компилятор макросов применяется к самому себе в режиме начальной загрузки, чтобы создать скомпилированную и гораздо более эффективную версию самого себя. Преимущество этого подхода заключается в том, что сложные приложения могут быть перенесены с одного компьютера на совершенно другой компьютер с очень небольшими усилиями (для каждой архитектуры целевой машины достаточно написать элементарный компилятор макросов). Появление современных языков программирования, особенно C, компиляторы которого доступны практически на всех компьютерах, сделало такой подход излишним. Однако это был один из первых (если не первый) случаев начальной загрузки компилятора.
Хотя макрокоманды могут быть определены программистом для любого набора программных инструкций на собственном ассемблере, обычно макросы связаны с библиотеками макросов, поставляемыми с операционной системой, что позволяет получить доступ к функциям операционной системы, таким как
В более старых операционных системах, таких как те, которые используются на мэйнфреймах IBM, полная функциональность операционной системы была доступна только для программ на языке ассемблера, а не для программ на языке высокого уровня (если, конечно, не использовались подпрограммы на языке ассемблера), поскольку стандартные макрокоманды не всегда иметь аналоги в подпрограммах, доступных для языков высокого уровня.
В середине 1950-х годов, когда программирование на языке ассемблера широко использовалось для написания программ для цифровых компьютеров, использование макросов было инициировано для двух основных целей: уменьшить объем программного кода, который должен был быть написан, путем генерации нескольких операторов языка ассемблера. из одной макрос-инструкции и для обеспечения соблюдения стандартов написания программ, например, для определения команд ввода / вывода стандартными способами. Макроинструкции были фактически промежуточным звеном между программированием на языке ассемблера и последующими языками программирования высокого уровня, такими как FORTRAN и COBOL. Две из самых первых программных установок для разработки «макроязыков» для компьютера IBM 705 были в Dow Chemical Corp. в Делавэре и в Управлении авиационных материалов Управления логистики баллистических ракет в Калифорнии. Макро-инструкция, написанная в формате целевого языка ассемблера, будет обрабатываться компилятором макросов, который был препроцессором ассемблера, чтобы сгенерировать одну или несколько инструкций на языке ассемблера, которые затем будут обработаны программой ассемблера, которая будет переводить инструкции на языке ассемблера в инструкции на машинном языке.
К концу 1950-х за макроязыком последовали Macro Assemblers. Это была комбинация того, что одна программа выполняла обе функции, макропрепроцессора и ассемблера в одном пакете.
В 1959 году Дуглас Иствуд и Дуглас Макилрой из Bell Labs ввели условные и рекурсивные макросы в популярный ассемблер SAP, создав так называемый Macro SAP. Статья Макилроя 1960 года была плодотворной в области расширения любых (в том числе высокоуровневых ) языков программирования с помощью макропроцессоров.
Макроассемблеры позволили программистам на ассемблере реализовать свой собственный макроязык и ограничили переносимость кода между двумя машинами, на которых работает один и тот же процессор, но с разными операционными системами, например ранние версии MSDOS и CPM-86. Библиотека макросов должна быть написана для каждой целевой машины, но не для всей программы на языке ассемблера. Обратите внимание, что более мощные макроассемблеры позволяли использовать конструкции условной сборки в макросах, которые могли генерировать разный код на разных машинах или в разных операционных системах, что уменьшало потребность в нескольких библиотеках.
В 1980-х и начале 1990-х настольные ПК работали только на нескольких МГц, и процедуры на языке ассемблера обычно использовались для ускорения программ, написанных на C, Fortran, Pascal и других. В то время в этих языках использовались разные соглашения о вызовах. Макросы можно использовать для сопряжения подпрограмм, написанных на языке ассемблера, с интерфейсной частью приложений, написанных практически на любом языке. Опять же, основной код языка ассемблера остался прежним, только библиотеки макросов нужно было написать для каждого целевого языка.
В современных операционных системах, таких как Unix и его производные, доступ к операционной системе обеспечивается с помощью подпрограмм, обычно предоставляемых динамическими библиотеками. Языки высокого уровня, такие как C, предлагают полный доступ к функциям операционной системы, устраняя необходимость в программах на языке ассемблера для таких функций.