Пакетный файл — это файл сценария в DOS , OS/2 и Microsoft Windows . Он состоит из серии команд , которые должны выполняться интерпретатором командной строки и храниться в текстовом файле. Пакетный файл может содержать любую команду, которую интерпретатор принимает в интерактивном режиме, и использовать конструкции, обеспечивающие условное ветвление и цикл внутри пакетного файла, например IF
, FOR
и GOTO
labels . Термин «пакетный» происходит от пакетной обработки , что означает «неинтерактивное выполнение», хотя пакетный файл может не обрабатывать пакет из нескольких данных.
Подобно языку управления заданиями (JCL), DCL и другим системам на мэйнфреймах и миникомпьютерах, пакетные файлы были добавлены для облегчения работы, необходимой для определенных регулярных задач, позволяя пользователю настроить сценарий для их автоматизации. При запуске пакетного файла программа оболочки (обычно COMMAND.COM или cmd.exe ) читает файл и выполняет его команды, обычно построчно. [1] Unix-подобные операционные системы , такие как Linux , имеют аналогичный, но более гибкий тип файла, называемый сценарием оболочки . [2]
Расширение имени файла .bat используется в DOS и Windows. В Windows NT и OS/2 также добавлен .cmd . Пакетные файлы для других сред могут иметь другие расширения, например, .btm в оболочках, связанных с 4DOS , 4OS2 и 4NT .
Детальная обработка пакетных файлов существенно изменилась в разных версиях. Некоторые сведения в этой статье применимы ко всем пакетным файлам, а другие сведения относятся только к определенным версиям.
В MS-DOS пакетный файл можно запустить из интерфейса командной строки, введя его имя, а затем любые необходимые параметры и нажав клавишу ↵ Enter. При загрузке DOS файл AUTOEXEC.BAT , если он присутствует, автоматически выполняется, поэтому любые команды, которые необходимо выполнить для настройки среды DOS, могут быть помещены в этот файл. Пользователи компьютеров могут использовать файл AUTOEXEC.BAT для установки системной даты и времени, инициализации среды DOS, загрузки любых резидентных программ или драйверов устройств или инициализации сетевых подключений и назначений.
Расширение имени файла .bat идентифицирует файл, содержащий команды, которые выполняются интерпретатором команд COMMAND.COM построчно, как если бы это был список команд, введенных вручную, с некоторыми дополнительными командами, специфичными для пакетного файла, для основных функций программирования. включая GOTO
команду изменения потока выполнения строки.
Microsoft Windows была представлена в 1985 году как надстройка на основе графического пользовательского интерфейса (GUI) к текстовым операционным системам и была разработана для работы в DOS. Для его запуска WIN
использовалась команда, которую можно было добавить в конец файла AUTOEXEC.BAT , чтобы разрешить автоматическую загрузку Windows. В более ранних версиях можно было запустить файл типа .bat из Windows в командной строке MS-DOS. Windows 3.1x и более ранние версии, а также Windows 9x вызывали COMMAND.COM для запуска пакетных файлов.
Операционная система IBM OS/2 поддерживала пакетные файлы в стиле DOS. Он также включал версию REXX , более продвинутого языка сценариев пакетных файлов . IBM и Microsoft начали разработку этой системы, но во время ее создания разошлись из-за спора; в результате IBM назвала свою DOS-подобную консольную оболочку без упоминания Microsoft, назвав ее просто DOS, хотя это, по-видимому, не имело никакого значения в отношении работы пакетных файлов из COMMAND.COM.
Интерпретатор пакетных файлов OS/2 также поддерживает команду EXTPROC. При этом пакетный файл передается программе, указанной в файле EXTPROC, как файл данных. Названная программа может быть файлом сценария; это похоже на #! механизм, используемый Unix-подобными операционными системами.
В отличие от Windows 98 и более ранних версий, семейство операционных систем Windows NT не зависит от MS-DOS. Windows NT представила улучшенный 32-битный интерпретатор команд ( cmd.exe ), который мог выполнять сценарии с расширением .CMD или .BAT. В Cmd.exe добавлены дополнительные команды и реализованы существующие немного по-другому, так что один и тот же командный файл (с другим расширением) может по-разному работать с cmd.exe и COMMAND.COM. В большинстве случаев операция идентична, если не используются несколько неподдерживаемых команд. Расширения Cmd.exe для COMMAND.COM можно отключить для совместимости.
Microsoft выпустила версию cmd.exe для Windows 9x и ME под названием WIN95CMD, чтобы позволить пользователям более старых версий Windows использовать определенные пакетные файлы в стиле cmd.exe.
Начиная с Windows 8 [обновлять], cmd.exe является обычным интерпретатором команд для пакетных файлов; более старый COMMAND.COM также можно запускать в 32-битных версиях Windows, способных запускать 16-битные программы. [номер 1]
append
, dpath
, ftype
, set
, path
и при выполнении из файла .bat изменяют значение переменной только в случае ошибки, тогда как из файла .cmd они влияют на уровень ошибки даже при возврате без ошибки assoc
. [3] Он также используется IBM OS/2 для пакетных файлов.prompt
errorlevel
COMMAND.COM и cmd.exe поддерживают специальные переменные ( %0
от %1
) %9
, чтобы ссылаться на путь и имя пакетного задания , а также на первые девять параметров вызова из пакетного задания, см. также SHIFT. Несуществующие параметры заменяются строкой нулевой длины. Их можно использовать аналогично переменным среды , но они не сохраняются в среде. Microsoft и IBM называют эти переменные параметрами замены или заменяемыми параметрами , тогда как Digital Research, Novell и Caldera ввели для них термин « переменные замены» [5] . JP Software называет их параметрами пакетного файла . [6]
В этом примере пакетный файл отображает Hello World!
, запрашивает и ждет, пока пользователь нажмет клавишу, а затем завершает работу. (Примечание: не имеет значения, написаны ли команды строчными или прописными буквами, если только вы не работаете с переменными)
@ ECHO OFF ECHO Привет, мир! ПАУЗА
Чтобы выполнить файл, его необходимо сохранить с суффиксом расширения имени файла .bat (или .cmd для операционных систем типа Windows NT) в обычном текстовом формате, обычно создаваемом с помощью текстового редактора, такого как Microsoft Notepad , или текстового процессора , работающего в обычный текстовый режим.
При выполнении отображается следующее:
Привет, мир!Нажмите любую клавишу для продолжения . . .
Интерпретатор выполняет каждую строку по очереди, начиная с первой. Символ @
в начале любой строки не позволяет подсказке отображать эту команду во время ее выполнения. Команда ECHO OFF
отключает подсказку навсегда или до тех пор, пока она не будет включена снова. Комбинация @ECHO OFF
часто, как здесь, представляет собой первую строку командного файла, которая сама по себе не позволяет отображать какие-либо команды. Затем выполняется следующая строка и ECHO Hello World!
выводится команда Hello World!
. Следующая строка выполняется, и PAUSE
команда отображает Press any key to continue . . .
и приостанавливает выполнение сценария. После нажатия клавиши скрипт завершается, так как команд больше нет. В Windows, если сценарий выполняется из уже запущенного окна командной строки , окно остается открытым в командной строке, как в MS-DOS; в противном случае окно закрывается при завершении.
Расширения переменных подставляются в команду в текстовом виде, и, таким образом, переменные, которые ничего не содержат, просто исчезают из синтаксиса, а переменные, содержащие пробелы, превращаются в несколько токенов. Это может привести к синтаксическим ошибкам или ошибкам.
Например, если %foo% пуст, этот оператор:
IF %foo% == bar ECHO Равно
анализирует как ошибочную конструкцию:
IF ==bar ECHO Равно
Аналогично, если %foo%
contains abc def
, возникает другая синтаксическая ошибка:
IF abc def == bar ECHO Равно
Обычный способ предотвратить эту проблему — заключить расширение переменных в кавычки, чтобы пустая переменная расширялась до допустимого выражения, IF ""=="bar"
а не до недопустимого IF ==bar
. Текст, который сравнивается с переменной, также должен быть заключен в кавычки, поскольку кавычки не являются специальным синтаксисом-разделителем; эти персонажи представляют собой самих себя.
IF " %foo% " == "bar" ECHO Равно
Задержка !VARIABLE! Расширение, доступное в Windows 2000 и более поздних версиях, можно использовать, чтобы избежать этих синтаксических ошибок. В этом случае переменные со значением NULL или переменные, состоящие из нескольких слов, не являются синтаксически ошибочными, поскольку значение расширяется после анализа команды IF:
ЕСЛИ !фу! == полоса ECHO Равно
Еще одно отличие Windows 2000 и более поздних версий заключается в том, что пустая переменная (неопределенная) не заменяется. Как описано в предыдущих примерах, предыдущее поведение пакетного интерпретатора приводило к пустой строке. Пример:
C:\> set MyVar = C:\> echo %MyVar% %MyVar%C:\> if " %MyVar% " == "" ( echo MyVar не определен ) else ( echo MyVar — %MyVar% ) MyVar — %MyVar%
Пакетные интерпретаторы до Windows 2000 отображали результат MyVar is not defined
.
В отличие от процессов Unix/POSIX, которые получают аргументы командной строки, уже разделенные оболочкой на массив строк, процесс Windows получает всю командную строку как одну строку через функцию API GetCommandLine. В результате каждое приложение Windows может реализовать собственный синтаксический анализатор для разделения всей командной строки на аргументы. Многие приложения и инструменты командной строки разработали для этого собственный синтаксис, поэтому не существует единого соглашения по заключению в кавычки или экранированию метасимволов в командных строках Windows.
cmd.exe
и wscript.exe
, используют свои собственные правила. [8]Если строка содержит кавычки и должна быть вставлена в другую строку текста, которая также должна быть заключена в кавычки, требуется особое внимание к механизму кавычек:
C:\> set foo = "эта строка заключена в кавычки"C:\> echo "test 1 %foo% " "test 1 "эта строка заключена в кавычки""C:\> eventcreate /T Предупреждение /ID 1 /L Система /SO «Источник» /D «Пример: %foo% » ОШИБКА: неверный аргумент/опция — «строка». Введите «EVENTCREATE /?» для использования.
В Windows 2000 и более поздних версиях решение состоит в том, чтобы заменить каждое появление символа кавычки внутри значения серией из трех символов кавычек:
C:\> set foo = "эта строка заключена в кавычки"C:\> set foo = %foo:"="""%C:\> echo "test 1 %foo% " "test 1 """эта строка заключена в кавычки""""C:\> eventcreate /T Предупреждение /ID 1 /L Система /SO «Источник» /D «Пример: %foo% » УСПЕХ: В журнале/источнике «Источник» создается событие типа «Предупреждение».
Некоторые символы, например |
символы вертикальной черты ( ), имеют особое значение в командной строке. Их нельзя распечатать как текст с помощью команды ECHO , если они не экранированы символом курсора ^:
C:\> echo foo | bar 'bar' не распознается как внутренняя или внешняя команда, исполняемая программа или командный файл.C:\> echo foo ^| бар фу | бар
Однако экранирование не работает должным образом при вставке экранированного символа в переменную среды. При простом отображении переменная содержит команду живого канала. Для отображения символа в виде текста в переменной необходимо экранировать как саму каретку, так и экранированный символ:
C:\> set foo = bar | baz 'baz' не распознается как внутренняя или внешняя команда, исполняемая программа или командный файл.C:\> set foo = bar ^| baz C:\> echo %foo% 'baz' не распознается как внутренняя или внешняя команда, исполняемая программа или командный файл.C:\> set foo = bar ^^^| baz C:\> echo %foo% bar | Баз
Отложенное раскрытие, доступное с помощью или с в Windows 2000 и более поздних версиях, может использоваться для отображения специальных символов, хранящихся в переменных среды, поскольку значение переменной расширяется после анализа команды:
C:\> cmd /V:ON Microsoft Windows [Версия 6.1.7601] Авторские права (c) принадлежат корпорации Microsoft, 2009 г. Все права защищены.C:\> set foo = bar ^| baz C:\> echo !foo! бар | Баз
До тех пор, пока в Windows Vista не появилась команда TIMEOUT, не существовало простого способа реализовать временную паузу, поскольку команда PAUSE останавливает активность сценария на неопределенный срок до тех пор, пока не будет нажата любая клавиша.
Было возможно множество обходных путей [10] , но обычно они работали только в некоторых средах: CHOICE
команда не была доступна в старых версиях DOS, PING
была доступна только в том случае, если был установлен TCP/IP и так далее. Microsoft не предоставила никакого решения, но ряд небольших служебных программ можно было установить из других источников. Коммерческим примером может служить команда Norton Utilities Batch Enhancer (BE) 1988 года, которая BE DELAY 18
ждала 1 секунду, или бесплатный 94-байтовый WAIT.COM [11] , где WAIT 5
ждал 5 секунд, а затем возвращал управление сценарию. Большинство таких программ представляют собой 16-разрядные файлы .COM, поэтому несовместимы с 64-разрядной версией Windows.
Обычно ко всему печатному тексту в конце каждой строки автоматически добавляются управляющие символы возврата каретки (CR) и перевода строки (LF).
@ echo foo @ echo bar
C:\> patchtest.bat foo bar
Не имеет значения, используют ли две команды echo одну и ту же командную строку; коды CR/LF вставляются, чтобы разбить вывод на отдельные строки:
C:\> @ echo Сообщение 1 &@ echo Сообщение 2 Сообщение 1 Сообщение 2
Хитрость, обнаруженная в Windows 2000 и более поздних версиях, заключается в использовании специального приглашения для ввода для вывода текста без CR/LF в конце текста. В этом примере CR/LF не следует за Сообщением 1, но следует за Строкой 2 и Строкой 3:
@ echo off set /p = «Сообщение 1» < nul echo Сообщение 2 echo Сообщение 3
C:\> patchtest2.bat Сообщение 1 Сообщение 2 Сообщение 3
Это можно использовать для вывода данных в текстовый файл без добавления CR/LF в конец:
C:\> set /p = «Сообщение 1» < nul > data.txt C:\> set /p = «Сообщение 2» < nul >> data.txt C:\> set /p = «Сообщение 3» < nul >> data.txt C:\> тип data.txt Сообщение 1 Сообщение 2 Сообщение 3
Однако невозможно внедрить этот урезанный вывод подсказки CR/LF непосредственно в переменную среды.
Невозможно иметь командную строку, которая использует путь UNC в качестве текущего рабочего каталога; например\\server\share\directory\
Командная строка требует использования букв дисков для назначения рабочего каталога, что усложняет запуск сложных пакетных файлов, хранящихся на общем ресурсе UNC сервера. Хотя пакетный файл можно запустить по пути к файлу UNC, рабочим каталогом по умолчанию является C:\Windows\System32\
.
В Windows 2000 и более поздних версиях обходным путем является использование команды PUSHD
и POPDс расширениями команд. [номер 2]
Если они не включены по умолчанию, расширения команд можно временно включить с помощью /E:ON
переключателя интерпретатора команд.
Таким образом, чтобы запустить пакетный файл в общей папке UNC, назначить временную букву диска общей папке UNC и использовать общую папку UNC в качестве рабочего каталога командного файла, можно создать ярлык Windows, который выглядит следующим образом:
Атрибут рабочего каталога этого ярлыка игнорируется.
Это также решает проблему, связанную с контролем учетных записей пользователей (UAC) в Windows Vista и более поздних версиях. Когда администратор вошел в систему и UAC включен, и он пытается запустить командный файл от имени администратора с буквы сетевого диска, используя контекстное меню файла, вызываемое правой кнопкой мыши, операция неожиданно завершается неудачно. Это связано с тем, что контекст привилегированной учетной записи UAC с повышенными правами не имеет назначений букв сетевых дисков, и невозможно назначить буквы дисков для контекста с повышенными правами с помощью оболочки Explorer или сценариев входа в систему. Однако если создать ярлык для командного файла с помощью приведенной выше конструкции PUSHD
/ POPD
и использовать ярлык для запуска командного файла от имени администратора, временная буква диска будет создана и удалена в контексте учетной записи с повышенными правами, и пакетный файл будет работать правильно. .
Следующий синтаксис правильно расширяется до пути текущего пакетного сценария.
%~dp0
Пути по умолчанию UNC отключены по умолчанию, поскольку они приводили к сбою старых программ. [12]
Значение реестра Dword DisableUNCCheck
в HKEY_CURRENT_USER\Software\Microsoft\Command Processor
[12] позволяет использовать каталог по умолчанию в формате UNC. CD
Команда откажется изменяться, но поместит путь UNC в каталог по умолчанию с помощью ярлыка Cmd или с помощью команды «Пуск». ( C$
доля предназначена для администраторов).
Пакетные файлы используют набор символов OEM, определенный компьютером, например, кодовая страница 437 . Их части, отличные от ASCII, несовместимы с наборами символов Unicode или Windows , которые иначе используются в Windows, поэтому необходимо соблюдать осторожность. [13] Имена файлов, отличные от английского, работают только в том случае, если они введены через редактор, совместимый с набором символов DOS. Имена файлов с символами, не входящими в этот набор, не работают в пакетных файлах.
Чтобы получить командную строку с Unicode вместо кодовой страницы 437 или аналогичной, можно использовать команду cmd /U
. В такой командной строке будет работать командный файл с именами файлов в Юникоде. Также можно использовать cmd /U
для прямого выполнения команд с использованием Unicode в качестве набора символов. Например, cmd /U /C dir > files.txt
создает файл, содержащий список каталогов с правильными символами Windows в кодировке UTF-16 LE.
Как и любой другой язык программирования, пакетные файлы могут использоваться злонамеренно. Простые трояны и форк-бомбы создаются легко, а пакетные файлы могут выполнять своего рода отравление DNS , изменяя файл хостов . Возможны пакетные вирусы, которые также могут распространяться через USB-накопители с помощью функции автозапуска Windows . [14]
Следующая команда в пакетном файле удалит все данные в текущем каталоге (папке) — без предварительного запроса подтверждения:
дель /Q *.*
Эти три команды представляют собой простую бомбу-вилку , которая будет постоянно воспроизводить себя, истощая доступные системные ресурсы, замедляя или приводя к сбою системы:
: начало TOP "" %0 перейти в TOP
Командный процессор cmd.exe, который интерпретирует файлы .cmd, поддерживается во всех 32-разрядных версиях Windows до Windows 10 и 64-разрядных версиях до Windows 11. COMMAND.EXE, который интерпретирует файлы .BAT, поддерживался во всех 16 версиях. - и 32-разрядные версии как минимум до Windows 10. [nb 3]
Для Windows доступны другие, более поздние и более мощные языки сценариев. Однако перед их использованием необходимо установить интерпретатор языка сценариев:
Файлы сценариев запускаются, если введено имя файла без расширения. Существуют правила приоритета, управляющие интерпретацией, скажем, DoThis
существования DoThis.com
, DoThis.exe
, DoThis.bat
, DoThis.cmd
и т. д.; по умолчанию DoThis.com
имеет наивысший приоритет. Этот порядок по умолчанию может быть изменен в новых операционных системах с помощью настраиваемой пользователем переменной среды PATHEXT .
COMMAND.COM
в командной строке 32-разрядной версии Windows 7.Две последовательные двойные кавычки внутри региона inQuotes должны привести к буквальной двойной кавычке (парсер остается в регионе inQuotes). Это поведение не является частью спецификации code:ParseArgumentsIntoList, но совместимо с CRT и .NET Framework.