С наступающим Новым годом! Форум программистов, компьютерный форум, киберфорум
Наши страницы
Batch (CMD/BAT)
Войти
Регистрация
Восстановить пароль
 
Рейтинг 4.58/111: Рейтинг темы: голосов - 111, средняя оценка - 4.58
Dragokas
Эксперт WindowsАвтор FAQ
17031 / 7089 / 857
Регистрация: 25.12.2011
Сообщений: 10,914
Записей в блоге: 16
1

SetLocal и расширенная обработка команд в Windows Batch [Статья]

10.05.2013, 21:38. Просмотров 21849. Ответов 2
Метки нет (Все метки)

SetLocal и расширенная обработка команд

Часто встречали такую конструкцию ?
Bash
1
SetLocal EnableExtensions EnableDelayedExpansion
По своей сути она состоит из 3 совмещенных частей:

Bash
1
2
3
SetLocal &:: Локализация переменных
SetLocal EnableExtensions &:: Использование дополнительных ключей
SetLocal EnableDelayedExpansion &:: Раскрытие переменных через знаки (!)
SetLocal

Что делает?
SetLocal - означает, что все переменные, объявленные внутри Batch-файла, будут локальными и после выхода из него обнуляться.
Это также оберегает от случаев, когда Вы забываете задавать начальное значение переменной.

По-умолчанию, значение переменной при инициализации
для строкового типа = пустой строке,
для числового типа = 0.
Типы переменных.

Если Вы запускаете Batch-файл
- из CMD.exe (ПУСК -> Выполнить -> CMD -> {Enter})
- один Batch-файл из другого
то при повторном запуске без команды SetLocal в переменных останутся старые значения.

Область видимости переменной, заданной командой Set, является текущая среда интерпретатора, если батник запущен через нее (CMD.exe) (и внутри нет команды локализации SetLocal), или сессия самого батника, если двойным кликом по нему.
Цитата Сообщение от Smitis Посмотреть сообщение
А также дочерние процессы, запущенный после того, как переменная установлена.
Как задается?
Bash
1
SetLocal
Переменные, использованные после команды SetLocal, можно обнулить принудительно еще до выхода из Batch-файла командой:
Bash
1
EndLocal
EndLocal и знак & (амперсанд)*


Конструкция вида:
Bash
1
2
3
SetLocal
Set B=1
EndLocal& Set A=%B%
Позволяет обнулить переменную B, при этом в однострочной команде после EndLocal
переменная раскрываемая через % и все еще будет содержать старое значение, поэтому может быть "переброшена" через локаль.

В конструкции вида:
Bash
1
2
3
setLocal EnableDelayedExpansion
Set B=1
EndLocal& Set A=!B!
переменная A получит актуальное (обнуленное) значение переменной B,
поэтому способ "переброса" значения через локаль невозможно использовать внутри циклов.


Extensions - режим расширенной обработки команд.

Что делает?
Позволяет использовать дополнительные ключи встроенных команд.

Где используется?
Откройте справку по командам, например,
For /?
Set /?
Там в первых 10 строках найдете выражение:
"Если включена расширенная обработка команд... то можно использовать еще и такие ключи..."
chdir /?
"Если включена расширенная обработка команд... в таком случае поведение команды изменяется" (касается пробелов в пути).
...

Как включается?
1) Задается командой
Bash
1
SetLocal EnableExtensions
EnableExtensions - обычно включена в системе по-умолчанию.
Но может быть выключена принудительно через реестр, а также по-умолчанию, выключена в некоторых старых ОС.
http://technet.microsoft.com/en-us/library/cc959665.aspx
Перенаправить поток ошибок программы в переменную
Поэтому, ее желательно включать во все скрипты.

2) Через реестр (выше по ссылкам).
3) Через ключи CMD.exe, если бат-файл или команда запускается через него, а не напрямую.
Метод может использоваться в однострочных командах, а также методах Shell других ЯП.
Пример:
Bash
1
cmd /E:ON /C Commands
где под Commands подразумевается путь и имя к командному файлу (bat, cmd), цикл, команда, или перечень команд, составленных по правилам формирования однострочных команд Batch.

Может быть отключена командой:
Bash
1
SetLocal DisableExtensions
Практическая польза от отключения, как мне известно, никакой.

DelayedExpansion - режим отложенного расширения переменных среды.

Что делает?
EnableDelayedExpansion - означает возможность использовать восклицательные знаки (!variable!) для раскрытия значения переменной.

Цитата Сообщение от Elroir
В обычном случае, чтобы узнать значение переменной окружения, надо воспользоваться конструкцией %переменная%. Однако, если значение переменной меняется внутри цикла и читается там же, то для корректной работы батника, нужно использовать команду setlocal enabledelayedexpansion и вместо символа "%", использовать символ "!". Иначе, значение переменной в цикле будет всегда одним и тем же - таким, каким было до входа в цикл.
Обычно переменные раскрываются через знак %.
"Раскрывается" - означает, что мы получаем значение, которое хранит переменная.

Пример:
Bash
1
2
3
4
:: Инициализируем переменную и задаем ей значение "Кот"
set Animal=Кот
:: Выводим на экран значение переменной Animal
echo %Animal%
Необходимость использовать знаки (!) возникает:
  • внутри цикла
  • под скобками
  • в однострочной команде
*Здесь и далее под конструкцией подразумевается один из указанных выше вариантов.

Если Вы измените значение переменной внутри конструкции,
раскрывая переменную через %, Вы получите ее старое значение (присвоенное до входа в эту конструкцию).

Новое (актуальное) значение переменной можно получить:
  • раскрыв переменную через знаки (!)
  • временно выйдя за пределы конструкции (например, командой Call)
  • после выхода из данной конструкции (ее завершения)


Как включить возможность использовать (!):
По-умолчанию, данный режим выключен.

Включить можно:
1) командой:
Bash
1
SetLocal EnableDelayedExpansion
2) через реестр (на постоянной основе - пользоваться пп.1,3 будет не нужно, но будет работать только на Вашей машине.)
http://technet.microsoft.com/en-us/library/cc957361.aspx

3) запуском batch-файла или команд(ы) через специальный ключ CMD.exe:
Пример:
Bash
1
cmd /V:ON /C Commands
где под Commands подразумевается путь и имя к командному файлу (bat, cmd), цикл, команда, или перечень команд, составленных по правилам формирования однострочных команд Batch.

Почему бы всегда не использовать этот метод вместо % ?

1) раскрытие переменной через % быстрее, чем через !
Оптимизация кода

2) при чтении данных из файла, в содержимом которого есть знаки ! и последующей записью или выводом на экран
этой информации знаки ! "глотаются". Поэтому еще иногда используется команда отключения данного режима:
Bash
1
SetLocal DisableDelayedExpansion
3) Когда анализируешь свой же код, удобно понимать логику работы своего алгоритма: если видишь % под циклом, то сразу понятно, что значение этой переменной под циклом не изменяется, а если и изменяется, то в этой же конструкции новое значение не используется.


Примеры практического использования:


1) Цикл:
Bash
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
@echo off
SetLocal EnableExtensions EnableDelayedExpansion
 
:: Читаем файл example.txt
:: Заводим счетчик строк count
:: 5-ю строку файла выводим на экран
 
set count=0
 
For /F "delims=" %%a in (example.txt) do (
  set /A count+=1
  if !count!==5 echo %%a
  echo В этом месте значение count всегда будет = 0. Count = %count%
  echo А здесь мы получим актуальное значение Count = !Count!
)
2) Просто под скобками:
Bash
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
@echo off
SetLocal EnableDelayedExpansion
(
  rem Присвоим значение переменной
  set variable=text
 
  echo Формируем
  echo несколько
  echo строк
  echo текстового
  echo файла
 
  echo Здесь ничего не получим. Перед скобками переменной не было присвоено значение.
  echo.%variable%
 
  echo Здесь получим актуальное значение
  echo !variable!
) >> example.txt
3) В однострочных командах:

Bash
1
2
3
4
5
6
7
8
@echo off
SetLocal EnableDelayedExpansion
 
set st1=staraya stroka
set st1=novaya stroka& echo Здесь получим старую строку %st1%
 
set st2=staraya stroka
set st2=novaya stroka& echo Здесь получим новую строку !st2!



Ремарка:
Принцип раскрытия переменных влияет также на изменяемые глобальные переменные, например,
time и date, которые постоянно обновляются и отображают дату и время.

После входа в конструкцию эти переменные "замораживаются".
Чтобы получить актуальную дату/время необходимо раскрытие через знаки (!):

Bash
1
echo !date! !time!
6
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
10.05.2013, 21:38
Ответы с готовыми решениями:

Батник переименования файлов и "SetLocal и расширенная обработка команд"
Здравствуйте. пишу код переименования файлов. не могу разобраться с...

Можно ли организовать выполнение команд Telnet с использованием VBS и Batch?
Есть компьютер на винде 7 и есть компьютер на линуксе. Нужно проверить работает...

[Статья] Windows 10 Creators Update: что нового в Bash/WSL и Windows Console
Когда вышел Windows 10 Anniversary Update (AU), подсистема Windows Subsystem...

Windows 7 домашняя расширенная не подключается к инету?
В сети 2 домена на одном из них интернет, ноут подключается к сети через...

Как сделать Windows Batch file?
Не могу понять как это сделать Windows Batch file вот на этой странице...

2
Dragokas
Эксперт WindowsАвтор FAQ
17031 / 7089 / 857
Регистрация: 25.12.2011
Сообщений: 10,914
Записей в блоге: 16
17.02.2016, 20:04  [ТС] 2
Практические примеры к SetLocal EnableDelayedExpansion от ComSpec:

Комментарии в bat-файле
1
Dragokas
Эксперт WindowsАвтор FAQ
17031 / 7089 / 857
Регистрация: 25.12.2011
Сообщений: 10,914
Записей в блоге: 16
18.10.2017, 14:17  [ТС] 3
Поддержка режимов EnableExtensions и DelayedExpansion в старых версиях ОС.
0
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
18.10.2017, 14:17

Перевести код с Bash (Linux) на Batch (Windows)
возникла потребность компилировать свой свофт под ОС Windows, есть написанный...

Разметка диска утилитой diskpart в процессе установки Windows 7 x64 на новый ПК с UEFI BIOS [Статья]
*ред: оригинальное название: "Разметка диска и установка Windows 7 x64 на новый...

Для чего требуется setlocal enabledelayedexpansion ?
Объясните русским языком, зачем нужен setlocal enabledelayedexpansion ?


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
3
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2018, vBulletin Solutions, Inc.
Рейтинг@Mail.ru