1 | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
Циклические операции и примеры (команда FOR) * [Статья]04.02.2013, 00:56. Показов 391965. Ответов 16
Метки нет (Все метки)
Вдохновившись сообщением от FraidZZ, написал мини-статейку, основанную на изложенных им положениях.
Заметки Циклическиe операции FOR Командой FOR задаётся список команд, которые выполняются с КАЖДЫМ элементом набора. Набор* пишется внутри IN (...) Список команд пишется внутри DO (...) Командная строка выполняет эти команды раз за разом, при этом текущий элемент набора находится в переменной, заданной после %% (назовём ее переменной цикла). Например,
* Примеры под спойлером Виды наборов для FOR
Для команды For без ключей набором может являться: 1) Маска файлов* (или путь + маска файлов) - в двойных кавычках, или без них: IN (*.txt) Результат: список файлов с расширением .txt в текущем каталоге. IN (*.txt *.bat) Результат: список файлов с расширениеми .txt и .bat в текущем каталоге. IN ("C:\Folder 1\Doc_31-12-*.txt") Результат: тот же. Но поиск ведется в каталоге C:\Folder 1 (заметьте с пробелом в имени); имя файла начинается на Doc_31-12- Прим.: FOR без ключа не умеет выводить список каталогов. * маска файлов - это набор файлов, заданный с помощью подстановочных знаков * и/или ? где * - обозначает 0 или больше любых символов в имени файла. а ? - означает 0 или 1 любой символ в имени файла. 2) Строка - в двойных кавычках, или без них: Строкой считается любая последовательность символов, если она не содержит знаков маски * или ? Смысл цикла здесь в том, чтобы разбить такую строку по пробелам (или знакам табуляции) и выполнить с каждой подстрокой список команд.
*О других модификаторах переменной цикла можно почитать здесь и здесь. Не по теме: 3) еще есть смешанный тип. Это когда в наборе стоит маска (1-й описанный тип), а через пробел Строка (2 тип)... ведет себя вполне ожидаемо, но вряд ли найдет себе применение. О наборах для FOR с ключем /F далее в нижнем спойлере. Ключ /F часто используется для построчного разбора файла, т.е.
UseBackQ (Use back quotes) означает, что набор с двойными кавычками* подразумевает передачу в цикл имени файла. delims= означает, что в переменную %%a будет записана вся строка (без разделения по пробелу или знаку табуляции, т.к. стандартный разделитель заменен на NULL (пустой символ). В такой вариации: tokens=* приводит к тому же результату, что и delims=. Означает прекратить разбивку по разделителю после "0-го" токена, т.е. сразу же. * Этот вариант необходим для работы с файлом, путь или имя которого содержит пробелы. Можно было не использовать UseBackQ, тогда команда приняла бы вид:
** Примеры под спойлером Виды наборов для FOR /F
В цикле FOR /F вид задаваемого набора зависит от формы кавычек в IN (...), а также наличия ключевого слова UseBackQ Виды наборов для FOR /F: 1) Набор файлов (задание маски недопустимо!) без UseBackQ - задается без кавычек IN (...) __с UseBackQ - может задаваться как в кавычках так и без них. IN (...) IN ("...") Специфика работы
Функционал: чтение содержимого файла(ов) построчно в переменную цикла! Принцип работы: источником для разбиения по разделителю (delims) является содержимое файла, заданного внутри IN (...) или файлов, если они заданы через пробел. Исключение: принятый по-умолчанию разделитель (пробел и знак табуляции) для этой конструкции цикла не применяется. А что получится, если установить delims= (возле равно - знак пробела) ?
. 2) Строка (допускаются практически любые символы) без UseBackQ - задается с двойными кавычками IN ("...") __с UseBackQ - задается с одиночными прямыми кавычками IN ('...') 3) Команда (сначала выполняется она, а уже ее результаты обрабатываются циклом как строка(-и)) без UseBackQ - задается с одиночными прямыми кавычками IN ('...') __с UseBackQ - задается с одиночными обратными кавычками IN (`...`) Примеры: 1.1. Чтение файла - Набор файлов
Прим.: Echo. - с точкой - это обход ошибки, чтобы можно было напечатать пустую строку, точнее строку с пробелами. 1.2. Чтение файла - Набор файлов + UseBackQ Получаем возможность использовать пробелы.
(заметьте, в имени папки есть пробел). 2.1. Строка
Сообщение от Результат
3.1. Команда
Вот что попадает под разбор циклу: Далее цикл разбирает каждую строку по пробелам и табуляции на подстроки (токены). На примере 1-й строки: 1-й токен (%%a)=29.12.2012 ...2-й токен (%%b)=15:16 ...3-й токен (%%c)=<DIR> ...4-й токен (%%d)=Favorites ... Результат через Echo выводится на экран:
Альтернативный вариант нужен с целью, если одна из выполняемых команд требует точно такие же кавычки как и ключевое слово IN (...). Дабы не пользоваться символом экранирования, просто применяем другие кавычки. Отличительной особенностью FOR /F является умение работать через токены*, а также поддержка дополнительных ключевых слов: 1) eol - знак комментария в начале строки (1-й символ). Т.е. строки с таким символом не будут обрабатываться. (по умолчанию, знак точки с запятой ; ) 2) skip - пропуск определенного кол-ва обрабатываемых строк от начала файла 3) delims - задать другой разделитель(-ли) (по умолчанию, пробел и знак табуляции) 4) tokens - количество получаемых токенов (подстрок) в теле цикла и пределы разбивки по разделителю. Также можно задать конкретный № токена, который попадет в первую переменную цикла. 5) usebackq - изменение правил использования кавычек внутри IN (...) Детальную справку можно получить, введя в консоль команду FOR /? * Токены - это подстроки, которые попадают в переменные цикла %% в каждой из итераций. Они получаются в результате разбивки строки, заданной в IN (...), по разделителю, заданному в Delims= (по умолчанию, пробел и знак табуляции). В отличие, от FOR без ключа, в FOR /F все токены (все подстроки одной строки) попадают сразу В ПЕРВУЮ ИТЕРАЦИЮ цикла. Они будут распределены по РАЗНЫМ переменным цикла, идущим в алфавитном порядке*, начиная с буквы, заданной после FOR /F %% Т.е.
* Макс. количество токенов и обход ограничения
Максимальное кол-во токенов составляет - 26, если начальным указать %%a либо %%A (регистр имеет значение) При этом переход с %%z в %%A не происходит. Остальная часть подстрок опускается. Можно проверить: Windows Batch file For /F "tokens=1,2" %%z IN ("1 2") do echo %%A Метод показал Anonymоus в теме Символ переноса строки в переменной окружения Алгоритм заключается в замене разделителя на пробел с одновременным заключением каждого токена в двойные кавычки. Далее строка разбирается обычным циклом FOR без ключа. Более универсальные конструкции на основе FOR /F для работы с файлами/папками строятся путем помещения в IN ('...') команды, результаты от выполнения которой уж затем обрабатываются циклом. Чаще всего это команда Dir. А теперь рассмотрим более сложный пример:
1.2. Чтение файла (сложный пример). Давайте возьмем сложный пример, и раскусим "крепкий орешек"
Сообщение от Содержимое файла 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
означает математическое перечисление чисел от первого числа (0), до третьего (6), с шагом, указанным вторым числом (2) в наборе IN (...). Указанная команда выведет:
16
|
04.02.2013, 00:56 | |
Ответы с готовыми решениями:
16
Циклические операции. Программа не выдает ни результата, ни ошибки Добрый день. Вероятность того что команда а и команда б забьет 2,3,4,5 голов Почему команда «dir>test.txt» работает, а команда «ftp /?>test.txt» — нет? Статья |
04.02.2013, 00:57 [ТС] | 2 | |||||||||||||||||||||||||||||||||||
Продолжение...
Особенности, которые распространяются на циклы FOR 0) Если нужно записать в наборе несколько команд, то используем: а) знаки амперсанда &, если мы хотим записать все в одну строку:
2) Если предполагается получение пустого токена, остальные токены смещаются. К примеру, имеем файл с содержимым:
Сообщение от file.txt
Сообщение от cmd
Сообщение от cmd
Это такие символы, как | = ^ < > &. А также % ! - если они используются не для раскрытия переменной, кроме случая с !, когда не включено удаленное расширение переменных. Сюда же входит знак кавычки, которая соответствует форме кавычки, применяемой для определения вида набора. 4) Регистр переменной цикла имеет значение. 5) Изменить вручную значение переменной цикла нельзя. Например, set %%A=param не даст эффекта. 6) После выхода из цикла переменная цикла уничтожается. 7) Внутри цикла нельзя использовать комментарии вида :: (будет критическая ошибка и "вылет" из пакетного файла) Вместо этого используем Rem. 8) Внутри цикла нельзя использовать метки (будет такой же "вылет"). 9) Тем не менее из цикла можно безвозвратно выйти на метку Goto metka а также выйти в подпрограмму и вернутся снова в цикл точно в ту же позицию через команду Call :metka В конце подпрограммы должна стоять команда Exit /B 10) Для получения значения обычной переменной внутри цикла в случае, если этой переменной было присвоено новое значение внутри тела этого же цикла, необходимо раскрывать переменную через знаки восклицания !variant! - это называется отложенное расширение переменных. Перед использованием такого способа, его нужно включить, задав в начале пакетного файла команду:
Точно таким же образом (через ! ) необходимо раскрывать переменную времени внутри цикла, т.к. иначе для цикла время "замерзает". Происходит это потому, что в цикл попадает т.н. "слепок" переменных среды. И работа с ними через % происходит уже как с копией данных, а не актуальными значениями. 11) Когда после IN указана команда, под разбор цикла попадают такие потоки: 1-й поток (StdOut) с 3-го по 9-й поток (User Stream) При этом 2-й поток (StdError) отображается на экране сразу после выполнения указанной в цикле команды в IN (...) 2-й поток можно занулить, обычным способом, не забыв экран:
Это же правило касается вызова цикла через Shell-подобные команды на других языках программирования. 13) Переменную цикла нельзя раскрыть внутри строки модификаторов другого цикла: for /f %%q in (quote) do for /f "usebackq tokens=1* delims=%%q" %%a in ("%~dp01.txt") do echo %%a%%b
12
|
25.05.2013, 01:50 [ТС] | 4 | |||||
Практические примеры:
1) Использование ключа /D (спасибо sov44) Переименовать файл в имя папки 2)
Можно также задавать конкретный путь перед маской. (спасибо pistol за найденный пример.)
1
|
05.01.2014, 22:00 [ТС] | 5 | |||||
3) Использование модификатора %~$ПУТЬ:I для поиска файла в папках, заданных переменной среды Path:
wmic.exe - наш искомый файл
Сообщение от help
3
|
05.11.2014, 01:39 [ТС] | 6 | |||||
* см. примечание от ComSpec.
1
|
3461 / 1999 / 635
Регистрация: 26.02.2014
Сообщений: 1,457
|
|
05.11.2014, 03:57 | 7 |
Dragokas, надо бы ещё исключить кавычки из перечня разделителей и написать, что парные двойные кавычки, наоборот, объединяют часть строки (или всю строку) вместе с перечисленными разделителями в одно "слово", и при этом сами сохраняются, и если к ним справа или слева примыкает группа символов без разделителей, то она также входит в состав этого "слова".
Исходя из сказанного, надо признать, что данный метод крайне ненадёжен для подсчёта слов (в обычном смысле понятия "слово") в реальных текстах.
2
|
28.04.2015, 20:24 [ТС] | 8 | |||||
Удаление лидирующих нулей
4
|
Smitis
|
|||||
29.04.2015, 00:11
#9
|
|||||
Не по теме: Удаление лидирующих нулей (дополнение)
1
|
3461 / 1999 / 635
Регистрация: 26.02.2014
Сообщений: 1,457
|
||||||
29.04.2015, 11:20 | 10 | |||||
Циклическая конструкция с использованием сочетания модификаторов
Пример: код:
2
|
3461 / 1999 / 635
Регистрация: 26.02.2014
Сообщений: 1,457
|
|||||||||||||||||||||
30.04.2015, 13:01 | 11 | ||||||||||||||||||||
Как известно, значение переменной цикла нельзя просто подставить в качестве модификатора в другой, вложенный, цикл:
Результатом такой подстановки будет вывод ошибки: Если же сделать так:
то ошибки не будет: Но как всё же осуществить корректный вывод во вложенном цикле? Если попробовать присвоить какой-нибудь переменной значение переменной первого цикла, а эатем запустить второй цикл с этой переменной в новом экземпляре интерпретатора с включённым отложенным расширением, то ошибка не исчезает: код:
результат: Оказалось, что это можно сделать проще, а именно просто запустить вложенный цикл в новом экземпляре интерпретатора: код:
результат:
6
|
24.08.2015, 15:48 [ТС] | 12 |
1
|
29.01.2016, 00:20 [ТС] | 13 |
Дополнение к предыдущей ссылке: Получение пустых значений токенов в цикле с командой FOR
0
|
3461 / 1999 / 635
Регистрация: 26.02.2014
Сообщений: 1,457
|
||||||
29.01.2016, 06:55 | 14 | |||||
Разбор строки с учётом "пустых токенов".
Данный код работает со всеми специальными символами, включая и . Единственный минус -- восклицательный знак не может использоваться в качестве разделителя (но только в качестве разделителя, наличие же его в анализируемой строке не вызовет проблем).
P.S. Вот как необычно непарная кавычка в строке код раскрасила.
3
|
7 / 5 / 0
Регистрация: 15.05.2015
Сообщений: 350
|
|
04.02.2016, 21:33 | 15 |
Dragokas, присвоение значения переменной - результатом выполнения внешнего исполнительного файла?
0
|
30.04.2017, 23:56 | 17 |
FOR %%a IN (C:\$\*.sfp) do dd if=C:\$\patch of="%%a" bs=1d
Я надеюсь без ошибок будет? Вроде получилось патчить файлы утилитой dd for windows!
0
|
30.04.2017, 23:56 | |
30.04.2017, 23:56 | |
Помогаю со студенческими работами здесь
17
Обработка прерывания "ошибочная команда" или "неверная команда" Статья о радиодальномере Статья о HYIP Случайная статья Статья по информатике Учебная статья Искать еще темы с ответами Или воспользуйтесь поиском по форуму: |