Присвоить вывод программы переменной с помощью командного файла MS

291

Мне нужно назначить вывод программы для переменной с помощью командного файла MS.

Так что в оболочке GNU Bash я бы использовал VAR=$(application arg0 arg1). Мне нужно подобное поведение в Windows, используя командный файл.

Нечто подобное set VAR=application arg0 arg1.

initialZero
источник

Ответы:

434

Одним из способов является:

application arg0 arg1 > temp.txt
set /p VAR=<temp.txt

Другой это:

for /f %%i in ('application arg0 arg1') do set VAR=%%i

Обратите внимание, что first %in %%iиспользуется для экранирования %после него и необходим при использовании приведенного выше кода в командном файле, а не в командной строке. Представь, у тебя test.batесть что-то вроде:

for /f %%i in ('c:\cygwin64\bin\date.exe +"%%Y%%m%%d%%H%%M%%S"') do set datetime=%%i
echo %datetime%
Карлос Гутьеррес
источник
11
Это отличный трюк, мне интересно, почему он не работает с трубкой
Билл К
25
Это работает только для вывода, который представляет собой одну строку текста (последующие строки опускаются после первого разрыва строки).
GroovyCakes
20
@Machta канал должен быть экранирован знаком ^ перед ним, внутри выражения в скобках. Пример:for /f "tokens=3" %%i in ('route print ^| findstr "\<0.0.0.0\>"') do set "myVar=%%i"
Эмануэле Дель Гранде
8
Не работайте для строки с пробелами. Например: для / f %% i in ('ver') установите VAR = %% i. Как писал @Renat, следует добавить «tokens = *»
Юра Шинкарев
2
@GroovyCakes Ответ на ваш вопрос о нескольких строках в ответе на повторяющийся вопрос
icc97
67

В дополнение к этому предыдущему ответу , внутри оператора for могут использоваться каналы, экранированные символом каретки:

    for /f "tokens=*" %%i in ('tasklist ^| grep "explorer"') do set VAR=%%i
Ренат
источник
1
Два важных момента: используйте жетоны, чтобы захватить и забрать, чтобы вырваться из трубы.
Кристофер Оезбек,
6
Эквивалентная версия, которая работает в интерфейсе командной строки и может быть вставлена ​​в копию для облегчения работы: for /f "tokens=*" %i in ('tasklist ^| findstr explorer') do @echo %iно в общем случае usebackqее следует использовать для обработки сложных команд.
Амит Найду
Токены были необходимы для обработки пробелов в выводе.
Марк Ингрэм
Котировки работать, а для меня, как это: for /f "tokens=*" %%i in ('"tasklist | grep explorer"') do set VAR=%%i. Мне проще, если в самой команде нет кавычек.
Пол
10

@OP, вы можете использовать циклы for для получения статуса возврата вашей программы, если она выводит что-то отличное от чисел

ghostdog74
источник
8

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

application arg0 arg1
set VAR=%errorlevel%
АКФ
источник
5
К сожалению, выводом является строка.
initialZero
хорошо. я оставлю это для потомков, но взгляну на ссылку @ jdigital, в которой говорится о передаче данных во временный файл.
2010 г.,
1
Вывод программы в stdout и stderr отличается от целочисленного возвращаемого значения. Программа может возвращать целочисленное значение, как в примере выше, а также отправлять строку на консоль (или перенаправлять в файл или в другое место). Они не являются взаимоисключающими и представляют собой две разные концепции.
Дэвид Ректор
7

При выполнении: for /f %%i in ('application arg0 arg1') do set VAR=%%iя получаю ошибку: %% я был неожиданным в это время. Как исправить, я должен был выполнить выше, какfor /f %i in ('application arg0 arg1') do set VAR=%i

Муниш Мехта
источник
9
В командном файле вам нужно %%и вне командного файла в командной строке вам нужно%
Джерри Иеремия
2

В дополнение к ответу вы не можете напрямую использовать операторы перенаправления вывода в заданной части forцикла (например, если вы хотите скрыть вывод stderror от пользователя и предоставить более приятное сообщение об ошибке). Вместо этого вы должны избегать их с помощью символа вставки ( ^):

for /f %%O in ('some-erroring-command 2^> nul') do (echo %%O)

Ссылка: перенаправить вывод команды в цикл for batch script

Kubo2
источник
1
@echo off
SETLOCAL ENABLEDELAYEDEXPANSION

REM Prefer backtick usage for command output reading:
REM ENABLEDELAYEDEXPANSION is required for actualized
REM  outer variables within for's scope;
REM within for's scope, access to modified 
REM outer variable is done via !...! syntax.

SET CHP=C:\Windows\System32\chcp.com

FOR /F "usebackq tokens=1,2,3" %%i IN (`%CHP%`) DO (
    IF "%%i" == "Aktive" IF "%%j" == "Codepage:" (
        SET SELCP=%%k
        SET SELCP=!SELCP:~0,-1!
    )
)
echo actual codepage [%SELCP%]

ENDLOCAL
RCM
источник
(плюс1) для объяснения
Сандбург
1

Вы можете использовать пакетный макрос для простого захвата выходных данных команды, немного похожий на поведение оболочки bash.

Использование макроса просто и выглядит

%$set% VAR=application arg1 arg2

И это работает даже с трубами

%$set% allDrives="wmic logicaldisk get name /value | findstr "Name""

Макрос использует переменную как массив и сохраняет каждую строку в отдельном индексе.
В примере %$set% allDrives="wmic logicaldiskбудут созданы следующие переменные:

allDrives.Len=5
allDrives.Max=4
allDrives[0]=Name=C:
allDrives[1]=Name=D:
allDrives[2]=Name=F:
allDrives[3]=Name=G:
allDrives[4]=Name=Z:
allDrives=<contains the complete text with line feeds>

Чтобы использовать его, не важно понимать, как работает сам макрос.

Полный пример

@echo off
setlocal

call :initMacro

%$set% ipOutput="ipconfig"
call :ShowVariable ipOutput
echo First line is %ipOutput[0]%

echo( 
%$set% driveNames="wmic logicaldisk get name /value | findstr "Name""
call :ShowVariable driveNames

exit /b

:ShowVariable
setlocal EnableDelayedExpansion
for /L %%n in (0 1 !%~1.max!) do (
    echo %%n: !%~1[%%n]!
)
echo(
exit /b

:initMacro
if "!!"=="" (
    echo ERROR: Delayed Expansion must be disabled while defining macros
    (goto) 2>nul
    (goto) 2>nul
)
(set LF=^
%=empty=%
)
(set \n=^^^
%=empty=%
)

set $set=FOR /L %%N in (1 1 2) dO IF %%N==2 ( %\n%
    setlocal EnableDelayedExpansion                                 %\n%
    for /f "tokens=1,* delims== " %%1 in ("!argv!") do (            %\n%
        endlocal                                                    %\n%
        endlocal                                                    %\n%
        set "%%~1.Len=0"                                            %\n%
        set "%%~1="                                                 %\n%
        if "!!"=="" (                                               %\n%
            %= Used if delayed expansion is enabled =%              %\n%
                setlocal DisableDelayedExpansion                    %\n%
                for /F "delims=" %%O in ('"%%~2 | findstr /N ^^"') do ( %\n%
                if "!!" NEQ "" (                                    %\n%
                    endlocal                                        %\n%
                    )                                               %\n%
                setlocal DisableDelayedExpansion                    %\n%
                set "line=%%O"                                      %\n%
                setlocal EnableDelayedExpansion                     %\n%
                set pathExt=:                                       %\n%
                set path=;                                          %\n%
                set "line=!line:^=^^!"                              %\n%
                set "line=!line:"=q"^""!"                           %\n%
                call set "line=%%line:^!=q""^!%%"                   %\n%
                set "line=!line:q""=^!"                             %\n%
                set "line="!line:*:=!""                             %\n%
                for /F %%C in ("!%%~1.Len!") do (                   %\n%
                    FOR /F "delims=" %%L in ("!line!") Do (         %\n%
                        endlocal                                    %\n%
                        endlocal                                    %\n%
                        set "%%~1[%%C]=%%~L" !                      %\n%
                        if %%C == 0 (                               %\n%
                            set "%%~1=%%~L" !                       %\n%
                        ) ELSE (                                    %\n%
                            set "%%~1=!%%~1!!LF!%%~L" !             %\n%
                        )                                           %\n%
                    )                                               %\n%
                    set /a %%~1.Len+=1                              %\n%
                )                                                   %\n%
            )                                                       %\n%
        ) ELSE (                                                    %\n%
            %= Used if delayed expansion is disabled =%             %\n%
            for /F "delims=" %%O in ('"%%~2 | findstr /N ^^"') do ( %\n%
                setlocal DisableDelayedExpansion                    %\n%
                set "line=%%O"                                      %\n%
                setlocal EnableDelayedExpansion                     %\n%
                set "line="!line:*:=!""                             %\n%
                for /F %%C in ("!%%~1.Len!") DO (                   %\n%
                    FOR /F "delims=" %%L in ("!line!") DO (         %\n%
                        endlocal                                    %\n%
                        endlocal                                    %\n%
                        set "%%~1[%%C]=%%~L"                        %\n%
                    )                                               %\n%
                    set /a %%~1.Len+=1                              %\n%
                )                                                   %\n%
            )                                                       %\n%
        )                                                           %\n%
        set /a %%~1.Max=%%~1.Len-1                                  %\n%
)                                                                   %\n%
    ) else setlocal DisableDelayedExpansion^&set argv=

goto :eof
Джеб
источник
0

Я написал скрипт, который пингует google.com каждые 5 секунд и регистрирует результаты с текущим временем. Здесь вы можете найти выходные данные для переменных "commandLineStr" (с индексами)

@echo off

:LOOPSTART

echo %DATE:~0% %TIME:~0,8% >> Pingtest.log

SETLOCAL ENABLEDELAYEDEXPANSION
SET scriptCount=1
FOR /F "tokens=* USEBACKQ" %%F IN (`ping google.com -n 1`) DO (
  SET commandLineStr!scriptCount!=%%F
  SET /a scriptCount=!scriptCount!+1
)
@ECHO %commandLineStr1% >> PingTest.log
@ECHO %commandLineStr2% >> PingTest.log
ENDLOCAL

timeout 5 > nul

GOTO LOOPSTART
Ja Vy
источник