Bat файл цикл while

Bat файл цикл while thumbnail

I have a trick for doing this!

I came up with this method because while on the CLI, it’s not possible to use the methods provided in the other answers here and it had always bugged me.

I call this the “Do Until Break” or “Infinite” Loop:

Basic Example

FOR /L %L IN (0,0,1) DO @(
ECHO. Counter always 0, See “%L” = “0” – Waiting a split second&ping -n 1 127.0.0.1>NUL )

This is truly an infinite loop!

This is useful for monitoring something in a CMD window, and allows you to use CTRL+C to break it when you’re done.

Want to Have a counter?

Either use SET /A OR You can modify the FOR /L Loop to do the counting and still be infinite (Note, BOTH of these methods have a 32bit integer overflow)

SET /A Method:

FOR /L %L IN (0,0,1) DO @(
SET /A “#+=1″&ECHO. L Still equals 0, See “%L = 0”! – Waiting a split second &ping -n 1 127.0.0.1>NUL )

Native FOR /L Counter:

FOR /L %L IN (-2147483648,1,2147483648) DO @(
ECHO.Current value of L: %L – Waiting a split second &ping -n 1 127.0.0.1>NUL )

Counting Sets of 4294967295 and Showing Current Value of L:

FOR /L %L IN (1,1,2147483648) DO @(
(
IF %L EQU 0 SET /A “#+=1”>NUL
)&SET /A “#+=0″&ECHO. Sets of 4294967295 – Current value of L: %L – Waiting a split second &ping -n 1 127.0.0.1>NUL )

However, what if:

  • You’re interested in reviewing the output of the monitor, and concerned that I will pass the 9999 line buffer limit before I check it after it has completed.
  • You’d like to take some additional actions once the Thing I am monitoring is finished.

For this, I determined how to use a couple methods to break the FOR Loop prematurely effectively turning it into a “DO WHILE” or “DO UNTIL” Loop, which is otherwise sorely lacking in CMD.

NOTE: Most of the time a loop will continue to iterate past the condition you checked for, often this is a wanted behavior, but not in our case.

Turn the “infinite Loop” into a “DO WHILE” / “DO UNTIL” Loop

UPDATE: Due to wanting to use this code in CMD Scripts (and have them persist!) as well as CLI, and on thinking if there might be a “more Correct” method to achieve this I recommend using the New method!

New Method (Can be used inside CMD Scripts without exiting the script):

FOR /F %%A IN (‘
CMD /C “FOR /L %%L IN (0,1,2147483648) DO @( ECHO.%%L & IF /I %%L EQU 10 ( exit /b ) )”
‘) DO @(
ECHO %%~A
)

At CLI:

FOR /F %A IN (‘
CMD /C “FOR /L %L IN (0,1,2147483648) DO @( ECHO.%L & IF /I %L EQU 10 ( exit /b ) )”
‘) DO @(
ECHO %~A
)

Original Method (Will work on CLI just fine, but will kill a script.)

FOR /L %L IN (0,1,2147483648) DO @(
ECHO.Current value of L: %L – Waiting a split second &ping -n 1 127.0.0.1>NUL&(
IF /I %L EQU 10 (
ECHO.Breaking the Loop! Because We have matched the condition!&DIR >&0
)
)
) 2>NUL

Older Post Stuff

Through chance I had hit upon some ways to exit loops prematurely that did not close the CMD prompt when trying to do other things which gave me this Idea.

NOTE:

While ECHO.>&3 >NUL had worked for me in some scenarios, I have played with this off and on over the years and found that DIR >&0 >NUL was much more consistent.

I am re-writing this answer from here forward to use that method instead as I recently found the old note to myself to use this method instead.

Edited Post with better Method Follows:

DIR >&0 >NUL

The >NUL is optional, I just prefer not to have it output the error.

I prefer to match inLine when possible, as you can see in this sanitized example of a Command I use to monitor LUN Migrations on our VNX.

for /l %L IN (0,0,1) DO @(
ECHO.& ECHO.===========================================& (
[VNX CMD] | FINDSTR /R /C:”Source LU Name” /C:”State:” /C:”Time ” || DIR >&0 >NUL
) & Ping -n 10 1.1.1.1 -w 1000>NUL )

Also, I have another method I found in that note to myself which I just re-tested to confirm works just as well at the CLI as the other method.

Apparently, when I first posted here I posted an older iteration I was playing with instead of the two newer ones which work better:

In this method, we use EXIT /B to exit the For Loop, but we don’t want to exit the CLI so we wrap it in a CMD session:

FOR /F %A IN (‘CMD /C “FOR /L %L IN (0,1,10000000) DO @( ECHO.%L & IF /I %L EQU 10 ( exit /b ) )” ‘) DO @(ECHO %~A)

Because the loop itself happens in the CMD session, we can use EXIT /B to exit the iteration of the loop without losing our CMD Session, and without waiting for the loop to complete, much the same as with the other method.

I would go so far as to say that this method is likely the “intended” method for the sort of scenario where you want to break a for loop at the CLI, as using CMD session is also the only way to get Delayed expansion working at the CLI for your loops, and the behavior and such behavior is clearly an intended workflow to leave a CMD session.

IE: Microsoft clearly made an intentional effort to have CMD Exit /B For loops behave this way, while the “Intended” way of doing this, as my other method, relies on having accidentally created just the right error to kick you out of the loop without letting the loop finish processing, which I only happenstantially discovered, and seems to only reliably work when using the DIR command which is fairly strange.

So that said, I think it’s probably a better practice to use Method 2:

FOR /F %A IN (‘CMD /C “FOR /L %L IN (0,1,10000000) DO @( ECHO.%L & IF /I %L EQU 10 ( exit /b ) )” ‘) DO @(ECHO %~A)

Читайте также:  Головная боль перед циклом

Although I suspect Method 1 is going to be slightly faster:

FOR /L %L IN (0,1,10000000) DO @( ECHO.%L & IF /I %L EQU 10 ( DIR >&) >NUL ) )

And in either case, both should allow DO-While loops as you need for your purposes.

Источник

  • #1

Циклическиe операции FOR

Командой FOR задаётся список команд, которые выполняются с КАЖДЫМ элементом набора.
Набор* пишется внутри IN (…)
Список команд пишется внутри DO (…)
Командная строка выполняет эти команды раз за разом, при этом текущий элемент набора находится в переменной, заданной после %% (назовём ее переменной цикла).

Например,

FOR %%a IN (C:Users*.txt) DO echo %%a

выведет имена и путь ко всем файлам с расширением .txt в папке C:Users.

* Примеры под спойлером

Для команды For без ключей набором может являться:

1) Маска файлов* (или путь + маска файлов)
– в двойных кавычках, или без них:

IN (*.txt)
Результат: список файлов с расширением .txt в текущем каталоге.

IN (*.txt *.bat)
Результат: список файлов с расширениеми .txt и .bat в текущем каталоге.

IN (“C:Folder 1Doc_31-12-*.txt”)
Результат: тот же. Но поиск ведется в каталоге C:Folder 1 (заметьте с пробелом в имени);
имя файла начинается на Doc_31-12-

Прим.: FOR без ключа не умеет выводить список каталогов.
* маска файлов – это набор файлов, заданный с помощью подстановочных знаков * и/или ?
где * – обозначает 0 или больше любых символов в имени файла.
а ? – означает 0 или 1 любой символ в имени файла.

2) Строка
– в двойных кавычках, или без них:

Строкой считается любая последовательность символов, если она не содержит знаков маски * или ?
Смысл цикла здесь в том, чтобы разбить такую строку по пробелам (или знакам табуляции)
и выполнить с каждой подстрокой список команд.

FOR %%a IN (гитара рыбка) do echo Моя %%a

Результат:

Если мы хотим, чтобы какая-то из строк не “билась” по пробелам, укажем ее в двойных кавычках:

FOR %%a IN (гитара “дорогая рыбка”) do echo Моя %%~a

Результат:

Моя гитара
Моя дорогая рыбка

При этом, чтобы не выводились сами кавычки “” мы используем модификатор* ~ (тильда) при раскрытии переменной цикла %%~a

*О других модификаторах переменной цикла можно почитать здесь и здесь.

3) еще есть смешанный тип. Это когда в наборе стоит маска (1-й описанный тип), а через пробел Строка (2 тип)… ведет себя вполне ожидаемо, но вряд ли найдет себе применение.

О наборах для FOR с ключем /F далее в нижнем спойлере.

Ключ /F часто используется для построчного разбора файла, т.е.

FOR /f “usebackq delims=” %%a IN (“C:1.txt”) DO echo %%a

выведет все строки файла 1.txt, который находится в корне диска C.

UseBackQ (Use back quotes) означает, что набор с двойными кавычками* подразумевает передачу в цикл имени файла.
delims= означает, что в переменную %%a будет записана вся строка (без разделения по пробелу или знаку табуляции, т.к. стандартный разделитель заменен на NULL (пустой символ).
В такой вариации:
tokens=* приводит к тому же результату, что и delims=. Означает прекратить разбивку по разделителю после “0-го” токена, т.е. сразу же.

* Этот вариант необходим для работы с файлом, путь или имя которого содержит пробелы.
Можно было не использовать UseBackQ, тогда команда приняла бы вид:

FOR /f “tokens=” %%a IN (C:1.txt) DO echo %%a

но такая конструкция восприняла бы пробел в имени как определение нового файла, поэтому UseBackQ более приемлем.

** Примеры под спойлером

В цикле FOR /F вид задаваемого набора зависит от формы кавычек в IN (…), а также наличия ключевого слова UseBackQ

Виды наборов для FOR /F:
1) Набор файлов (задание маски недопустимо!)

без UseBackQ – задается без кавычек IN (…)
__с UseBackQ – может задаваться как в кавычках так и без них. IN (…) IN (“…”)

Функционал:

чтение содержимого файла(ов) построчно в переменную цикла!

Принцип работы:

источником для разбиения по разделителю (delims) является содержимое файла, заданного внутри IN (…) или файлов, если они заданы через пробел.

Исключение:

принятый по-умолчанию разделитель (пробел и знак табуляции) для этой конструкции цикла не применяется.

А что получится, если установить delims= (возле равно – знак пробела) ?

FOR /F “delims= ” %%a in (1.txt) do echo %%a

В выводе на экран командой Echo %%a мы получим содержимое файла, где каждая из строк будет напечатана до первого встретившегося пробела.

2) Строка (допускаются практически любые символы)

без UseBackQ – задается с двойными кавычками IN (“…”)
__с UseBackQ – задается с одиночными прямыми кавычками IN (‘…’)

3) Команда (сначала выполняется она, а уже ее результаты обрабатываются циклом как строка(-и))

без UseBackQ – задается с одиночными прямыми кавычками IN (‘…’)
__с UseBackQ – задается с одиночными обратными кавычками IN (`…`)

Примеры:

1.1. Чтение файла – Набор файлов

FOR /F %%a IN (c:users1.txt c:users2.txt) do echo.%%a

Результат: выведет подряд содержимое двух файлов – 1.txt и 2.txt из каталога c:users
Прим.: Echo. – с точкой – это обход ошибки, чтобы можно было напечатать пустую строку, точнее строку с пробелами.

1.2. Чтение файла – Набор файлов + UseBackQ
Получаем возможность использовать пробелы.

FOR /F “usebackq” %%a IN (“c:folder 11.txt”) do echo.%%a

Результат: выведет содержимое файла 1.txt из каталога c:folder 1
(заметьте, в имени папки есть пробел).

2.1. Строка

FOR /F %%a in (“Каждое слово в отдельную переменную”) echo a=%%a; b=%%b; c=%%c; d=%%d; e=%%e

Результат написал(а):

a=Каждое; b=слово; c=в; d=отдельную; e=переменную

2.2. Строка + UseBackQ

FOR /F “UseBackQ” %%a in (‘Каждое слово в отдельную переменную’) echo a=%%a; b=%%b; c=%%c; d=%%d; e=%%e

Результат такой же.

3.1. Команда

For /F “tokens=1-4” %%a IN (‘Dir /A:D-L’) Do Echo Папка %%d Дата модификации %%a. Время %%b

Сначала выполняется Dir /AD-L, которая выводит информацию о папках в текущем каталоге.
Вот что попадает под разбор циклу:

Далее цикл разбирает каждую строку по пробелам и табуляции на подстроки (токены).
На примере 1-й строки:
1-й токен (%%a)=29.12.2012 …2-й токен (%%b)=15:16 …3-й токен (%%c)=<DIR> …4-й токен (%%d)=Favorites

Результат через Echo выводится на экран:

Папка Favorites Дата модификации 29.12.2012. Время 15:16
Папка Links Дата модификации 14.01.2013. Время 01:51
…и т.д.

3.2. Команда + UseBackQ

For /F “Usebackq tokens=1-4” %%a IN (`Dir /A:D-L`) Do Echo Папка %%d Дата модификации %%a. Время %%b
Читайте также:  Отмена кок в начале цикла

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

Отличительной особенностью FOR /F является умение работать через токены*,
а также поддержка дополнительных ключевых слов:
1) eol – знак комментария в начале строки (1-й символ). Т.е. строки с таким символом не будут обрабатываться. (по умолчанию, знак точки с запятой ; )
2) skip – пропуск определенного кол-ва обрабатываемых строк от начала файла
3) delims – задать другой разделитель(-ли) (по умолчанию, пробел и знак табуляции)
4) tokens – количество получаемых токенов (подстрок) в теле цикла и пределы разбивки по разделителю.
Также можно задать конкретный № токена, который попадет в первую переменную цикла.
5) usebackq – изменение правил использования кавычек внутри IN (…)

Детальную справку можно получить, введя в консоль команду FOR /?

* Токены – это подстроки, которые попадают в переменные цикла %% в каждой из итераций.
Они получаются в результате разбивки строки, заданной в IN (…), по разделителю, заданному в Delims= (по умолчанию, пробел и знак табуляции).

В отличие, от FOR без ключа, в FOR /F все токены (все подстроки одной строки) попадают сразу В ПЕРВУЮ ИТЕРАЦИЮ цикла.
Они будут распределены по РАЗНЫМ переменным цикла, идущим в алфавитном порядке*, начиная с буквы, заданной после FOR /F %%

Т.е.

FOR /F %%a IN (“гитара рыбка”) do echo Моя %%a и моя %%b.

Результат:

*

Максимальное кол-во токенов составляет – 26,
если начальным указать %%a либо %%A (регистр имеет значение)
При этом переход с %%z в %%A не происходит. Остальная часть подстрок опускается.
Можно проверить:

For /F “tokens=1,2” %%z IN (“1 2”) do echo %%A

Бывают случаи, когда требуется разбить строку по специфическому разделителю и при этом выполнить одну и ту же команду над каждой из подстрок (токеном). Кол-во токенов неизвестно.
Метод показал Anonymоus в теме Символ переноса строки в переменной окружения

Алгоритм заключается в замене разделителя на пробел с одновременным заключением каждого токена в двойные кавычки. Далее строка разбирается обычным циклом FOR без ключа.

Более универсальные конструкции на основе FOR /F для работы с файлами/папками строятся путем помещения
в IN (‘…’) команды, результаты от выполнения которой уж затем обрабатываются циклом.
Чаще всего это команда Dir.

1.2. Чтение файла (сложный пример).
Давайте возьмем сложный пример, и раскусим “крепкий орешек”

FOR /F “UseBackQ skip=1 tokens=2 delims=/ eol=” %%X in (“%~dp01.txt”) do echo %%X

Имеем в распоряжении файл 1.txt, который находится рядом с батником.

Содержимое файла 1.txt написал(а):

первая строка
;комментарий/кода
наше любимоеблюдо

Порядок разбора (или “как прибл. будет думать ком. строка”):

1) %~dp0 – означает каталог, где находится батник, например c:temp

2) UseBackQ, ага – значит если в IN (…) ничего нет или двойные кавычки, то это файл и его нужно прочесть.

3) Читаем содержимое файла 1.txt в папке %~dp0, а затем каждую его строку проверяем по правилам… ключевых слов skip=1 tokens=2 delims=/ eol=

4) Итак, первая строка так и называется “первая строка”

skip=1 – означает пропустить от начала файла 1-у строку,
значит идем дальше:

5) Вторая строка: “;комментарий/кода”
Первый символ – это точка с запятой. А теперь смотрим сюда “eol=пустой символ”.
По умолчанию eol=; и если бы мы не указали пустой EOL, то цикл просто бы пропустил эту строку.
Итак, символ комментария не совпадает с заданным (т.е. он вообще пустой), а значит строчка нам подходит -> проверяем ее дальше:

6) Смотрим какие приняты разделители: delims=/ (знаки / и ), значит
из строки “;комментарий/кода”
мы получим 2 подстроки:
1-й токен – “;комментарий”
2-й токен – “кода”

7) Теперь смотрим сюда tokens=2 – значит, что первой букве цикла нужно присвоить значение 2-го токена.
Первая буква цикла у нас X. Переменная называется %%X
А второй токен – это подстрока “кода”

8) Только теперь мы попадаем в тело цикла: Echo %%X
что означает – вывести на экран текст “кода”

С 3-ей строкой потренируйтесь самостоятельно.

Ключ /L

FOR /L %%a IN (0,2,6) DO echo %%a

IN (первое, шаг, последнее)
означает математическое перечисление чисел от первого числа (0), до третьего (6), с шагом, указанным вторым числом (2) в наборе IN (…).

Указанная команда выведет:

Прим.: дробные числа командной строкой не поддерживаются.

  • #2

Продолжение…

Особенности, которые распространяются на циклы FOR

0) Если нужно записать в наборе несколько команд, то используем:

а) знаки амперсанда &, если мы хотим записать все в одну строку:

FOR /L %%a IN (1,1,10) DO echo Найден файл:& echo %%a

б) знаки скобок, если хотим разбить команды в несколько строк для наглядности и удобства:

FOR /L %%a IN (1,1,10) DO (
echo Найден файл:
echo %%a
)

1) Пустые строки в содержимом файлов и выводе команд игнорируются.

2) Если предполагается получение пустого токена, остальные токены смещаются.
К примеру,
имеем файл с содержимым:

file.txt написал(а):

Применим команду:

FOR /F “tokens=1,2,3 delims=/” %%a in (file.txt) do echo a=%%a; b=%%b; c=%%c

Результат будет:

cmd написал(а):

вместо ожидаемого

cmd написал(а):

3) Все специальные (служебные) символы необходимо экранировать (предварять птичкой ^).
Это такие символы, как | = ^ < > &. А также % ! – если они используются не для раскрытия переменной, кроме случая с !, когда не включено удаленное расширение переменных.
Сюда же входит знак кавычки, которая соответствует форме кавычки, применяемой для определения вида набора.

4) Регистр переменной цикла имеет значение.

5) Изменить вручную значение переменной цикла нельзя. Например, set %%A=param не даст эффекта.

6) После выхода из цикла переменная цикла уничтожается.

7) Внутри цикла можно использовать не более 1-го комментария вида :: (иначе, будет критическая ошибка и “вылет” из пакетного файла)
Вместо этого используем Rem.

8) Внутри цикла можно использовать не более 1-й метки (иначе, будет такой же “вылет”).

Пример максимально допустимой конструкции:

@echo off
for %%n in ( 0 1 2 3 4 5 6 7 8 9 ) do (
:label
echo %%n
:: comment
echo %%n
)
Читайте также:  Может ли овуляция быть позднее середины цикла

9) Тем не менее из цикла можно безвозвратно выйти на метку Goto metka
а также выйти в подпрограмму и вернутся снова в цикл точно в ту же позицию через команду Call :metka
В конце подпрограммы должна стоять команда Exit /B

10) Для получения значения обычной переменной внутри цикла в случае, если этой переменной было присвоено новое значение внутри тела этого же цикла, необходимо раскрывать переменную через знаки восклицания !variant! – это называется отложенное расширение переменных.
Перед использованием такого способа, его нужно включить, задав в начале пакетного файла команду:

SetLocal EnableDelayedExpansion

Иначе, если использовать обычный способ раскрытия %variant%, мы получим значение переменной, присвоенное ей еще до входа в цикл.
Точно таким же образом (через ! ) необходимо раскрывать переменную времени внутри цикла, т.к. иначе для цикла время “замерзает”.
Происходит это потому, что в цикл попадает т.н. “слепок” переменных среды. И работа с ними через % происходит уже как с копией данных, а не актуальными значениями.

11) Когда после IN указана команда, под разбор цикла попадают такие потоки:
1-й поток (StdOut)
с 3-го по 9-й поток (User Stream)

При этом 2-й поток (StdError) отображается на экране сразу после выполнения указанной в цикле команды в IN (…)
2-й поток можно занулить, обычным способом, не забыв экран:

FOR /F %%A IN (‘dir %someparam% 2^>nul’) do rem

Также можно перенаправить 2-й поток, чтобы он обрабатывался вместе с первым, вместо его вывода на экран:

FOR /F %%A IN (‘dir %someparam% 2^>^&1’) do rem

Собственно, чтобы наоборот, обрабатывать 2-й поток вместо первого, можно воспользоваться следующей конструкцией:

FOR /F %%A IN (‘dir %someparam% 1^>nul 2^>^&3’) do rem

12) При запуске цикла из консоли (cmd.exe) переменные цикла нужно указывать без удвоения знака процента %a.
Это же правило касается вызова цикла через Shell-подобные команды на других языках программирования.

13) Переменную цикла нельзя раскрыть внутри строки модификаторов другого цикла:

for /f %%q in (quote) do for /f “usebackq tokens=1* delims=%%q” %%a in (“%~dp01.txt”) do echo %%a%%b

но есть способ обхода от ComSpec:

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

for /f “tokens=2” %i in (“1 3 5”) do @for /f “tokens=%i” %j in (“1 3 5 7 9″) do @echo %j

Результатом такой подстановки будет вывод ошибки:
[quote]
Непредвиденное появление: %i”.
[/quote]

Если же сделать так:
[code]
for /f “tokens=2” %i in (“1 3 5”) do @set x=%i
for /f “tokens=%x%” %j in (“1 3 5 7 9”) do @echo %j

то ошибки не будет:

Но как всё же осуществить корректный вывод во вложенном цикле?

Если попробовать присвоить какой-нибудь переменной значение переменной первого цикла, а эатем запустить второй цикл с этой переменной в новом экземпляре интерпретатора с включённым отложенным расширением, то ошибка не исчезает:

for /f “tokens=2” %i in (“1 3 5”) do @set x=%i& cmd/v/c for /f “tokens=!x!” %j in (“1 3 5 7 9”) do @echo %j

результат:

Непредвиденное появление: !x!”.

Оказалось, что это можно сделать проще, а именно просто запустить вложенный цикл в новом экземпляре интерпретатора:

for /f “tokens=2” %i in (“1 3 5”) do @cmd/c for /f “tokens=%i” %j in (“1 3 5 7 9”) do @echo %j

результат:

Последнее редактирование: 28 Июн 2016

  • #3

Продолжение…

Практические примеры:

1) Использование ключа /D

(спасибо sov44) Переименовать файл в имя папки

2) первый по алфавиту файл

for /F %%i in (“*.txt”) do set FILE_NAME=%%~nxi

Берет первый по алфавиту файл, заданный маской *.txt, и сразу выходит из цикла.
Можно также задавать конкретный путь перед маской.
(спасибо pistol за найденный пример.)

3) Использование модификатора %~$ПУТЬ:I для поиска файла в папках, заданных переменной среды Path:

for %%a in (“wmic.exe”) do echo %%~s$PATH:a

где:
wmic.exe – наш искомый файл

help написал(а):

Переменная среды ПУТЬ (в примере PATH), и переменная %a
заменяется на полное имя первого найденного файла.
Если переменная ПУТЬ не определена или в результате поиска
не найден ни один файл, то этот модификатор заменяется на пустую строку.

4) Разбивка текста по разделителям

for %%i in (text) do echo %%i

где text – текст, разбивка на итерации которого происходит по знакам:

  • пробел
  • парные кавычки *
  • табуляция
  • запятая
  • точка с запятой
  • знак равенства.

* Примечание от ComSpec:

Надо бы ещё исключить кавычки из перечня разделителей и написать, что парные двойные кавычки, наоборот, объединяют часть строки (или всю строку) вместе с перечисленными разделителями в одно “слово”, и при этом сами сохраняются, и если к ним справа или слева примыкает группа символов без разделителей, то она также входит в состав этого “слова”.
Исходя из сказанного, надо признать, что данный метод крайне ненадёжен для подсчёта слов (в обычном смысле понятия “слово”) в реальных текстах.

5) Удаление лидирующих нулей

:: математические операции со значениями 08 и 09 будут приводить к ошибкам
set /a m=%date:~3,2%
:: поэтому для начала, удаляем лидирующие нули
set m=%date:~3,2%
for /f “tokens=* delims=0” %%j in (“%m%”) do set m=%%j
echo %m%

Дополнение от Smitis:

Если количество цифр в числе, в котором надо удалить лидирующие нули, известно заранее, то можно поступить так (для месяца из двух цифр)

:: В вычитаемом числе должно быть столько нулей, сколько цифр исходном числе.
:: Исходное число должно быть не длинее 9-ти разрядов.
set /a m=1%date:~3,2%-100
echo %m%

Можно сразу подставлять такую конструкцию в нужное выражение.

Добавление лидирующих нулей: https://safezone.cc/threads/cmd-dobavit-lidirujuschie-nuli.32940/

6) Распределение итераций цикла for без ключей.
Дополнение: Получение пустых значений токенов в цикле с командой FOR

7) Использование кавычки в роле разделителя
Автор: amel27

for /f tokens^=1^,2^,3^ delims^=^” %%a in (“1″”2″”3”) do echo %%a–%%b–%%с

8) Циклическая конструкция с использованием сочетания модификаторов
(от ComSpec)

“tokens=* delims=<символы>”
является универсальным “триммером” для удаления любых заявленных символов из начала строки до первого незаявленного символа.

Пример:

for /f “tokens=* delims=0123456789” %i in (“972630014380333075vfh658weghui72qcfgb3658eyfgcb7652qweuy8”) do @echo %i

результат:

vfh658weghui72qcfgb3658eyfgcb7652qweuy8

9) получение результата выполнения программы в переменную
(если результатат – 1 строка)

for /f “delims= eol=” %%a in (‘file.exe’) do set “var=%%a”

Последнее редактирование: 18 Мар 2019

Источник