Арифметика даты в сценариях cmd - PullRequest
9 голосов
/ 10 декабря 2008

Мне нужно написать скрипт для изменения имени файла с aDate.txt на bDate.txt, где:

  • aDate - текущая системная дата в формате ггггммдд и
  • bDate - текущая системная дата - 1 в формате ггггммдд .

У меня сейчас есть:

set yy=%date:~6,2%
set mm=%date:~3,2%
set dd=%date:~0,2%
if "%date:~6,1%"==" " set yy=0%yy:~1,1%
if "%date:~3,1%"==" " set mm=0%mm:~1,1%
if "%date:~0,1%"==" " set dd=0%dd:~1,1%
SET sys_date=20%yy%%mm%%dd%
ECHO %sys_date%
REM still have to do this bit properly
SET sys_date_yesterday=%sys_date%a
move %sys_date%.txt %sys_date_yesterday%.txt

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

Есть идеи?

Ответы [ 4 ]

4 голосов
/ 10 декабря 2008

Вы должны сделать это трудным путем. Я предлагаю использовать это решение от SteveGTR. Я копирую текст ниже, потому что, по крайней мере, я не всегда вижу решение на этом сайте.


Вот пакетный файл, который я разработал, чтобы вычесть любое количество дней из текущей даты. Он принимает параметр командной строки количества дней. По умолчанию 1 день (вчера):

@echo off

set yyyy=

set $tok=1-3
for /f "tokens=1 delims=.:/-, " %%u in ('date /t') do set $d1=%%u
if "%$d1:~0,1%" GTR "9" set $tok=2-4
for /f "tokens=%$tok% delims=.:/-, " %%u in ('date /t') do (
 for /f "skip=1 tokens=2-4 delims=/-,()." %%x in ('echo.^|date') do (
    set %%x=%%u
    set %%y=%%v
    set %%z=%%w
    set $d1=
    set $tok=))

if "%yyyy%"=="" set yyyy=%yy%
if /I %yyyy% LSS 100 set /A yyyy=2000 + 1%yyyy% - 100

set CurDate=%mm%/%dd%/%yyyy%

set dayCnt=%1

if "%dayCnt%"=="" set dayCnt=1

REM Substract your days here
set /A dd=1%dd% - 100 - %dayCnt%
set /A mm=1%mm% - 100

:CHKDAY

if /I %dd% GTR 0 goto DONE

set /A mm=%mm% - 1

if /I %mm% GTR 0 goto ADJUSTDAY

set /A mm=12
set /A yyyy=%yyyy% - 1

:ADJUSTDAY

if %mm%==1 goto SET31
if %mm%==2 goto LEAPCHK
if %mm%==3 goto SET31
if %mm%==4 goto SET30
if %mm%==5 goto SET31
if %mm%==6 goto SET30
if %mm%==7 goto SET31
if %mm%==8 goto SET31
if %mm%==9 goto SET30
if %mm%==10 goto SET31
if %mm%==11 goto SET30
REM ** Month 12 falls through

:SET31

set /A dd=31 + %dd%

goto CHKDAY

:SET30

set /A dd=30 + %dd%

goto CHKDAY

:LEAPCHK

set /A tt=%yyyy% %% 4

if not %tt%==0 goto SET28

set /A tt=%yyyy% %% 100

if not %tt%==0 goto SET29

set /A tt=%yyyy% %% 400

if %tt%==0 goto SET29

:SET28

set /A dd=28 + %dd%

goto CHKDAY

:SET29

set /A dd=29 + %dd%

goto CHKDAY

:DONE

if /I %mm% LSS 10 set mm=0%mm%
if /I %dd% LSS 10 set dd=0%dd%

echo Date %dayCnt% day(s) before %CurDate% is %mm%/%dd%/%yyyy%

Удачи, Стив

2 голосов
/ 22 июля 2016

Простое добавление или вычитание дней из даты с помощью пакетного сценария Windows

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

Для добавления: Вы также можете использовать этот скрипт для добавления количества дней к текущей дате, удалив минус (-) Символ из приведенного ниже пакетного сценария в подпрограмме :DynamicVBSScriptBuild, поэтому, когда вы видите этот ,-%MinusDay%,, вы просто удаляете символ минус, чтобы получить ,%MinusDay%, в каждой из этих строк, а теперь переменную MinusDay= значение будет равно количеству дней, которое вы хотите добавить.

Важное примечание: Кажется, что пять 9 (99999) - это предел для пакетного сценария при вычитании со значением MinusDays=. Также кажется, что шесть 9 (999999) - это ограничение на пакетный скрипт при добавлении со значением MinusDays=.

Пакетный скрипт

@ECHO ON

::// Minus days is the number of days to subtract from the CURRENT DAY i.e. 2 for minus 2 days or 99999 for minus 99999 days from when it's run
SET MinusDay=2

:: This calls the temp vbs script routine that will be used to set YYYY-MM-DD values for the subtracted days date you specify
CALL :DynamicVBSScriptBuild

FOR /F "TOKENS=*" %%A IN ('cscript//nologo "%YYYYTmpVBS%"') DO SET YYYY=%%A
FOR /F "TOKENS=*" %%A IN ('cscript//nologo "%MMTmpVBS%"') DO SET MM=%%A
FOR /F "TOKENS=*" %%A IN ('cscript//nologo "%DDTmpVBS%"') DO SET DD=%%A

::// Set your web server log file path in the below variable
SET WebServerLogPath=C:\WebServer\Logs

::// Set web server log file name where YYYY MM DD variables are set to the values after the day numbers setup above are subtracted
SET YYYY=%YYYY%
SET MM=%MM%
SET DD=%DD%

ECHO %YYYY%%MM%%DD%
PAUSE

GOTO EOF

:DynamicVBSScriptBuild
SET YYYYTmpVBS=%temp%\~tmp_yyyy.vbs
SET MMTmpVBS=%temp%\~tmp_mm.vbs
SET DDTmpVBS=%temp%\~tmp_dd.vbs
IF EXIST "%YYYYTmpVBS%" DEL /Q /F "%YYYYTmpVBS%"
IF EXIST "%MMTmpVBS%" DEL /Q /F "%MMTmpVBS%"
IF EXIST "%DDTmpVBS%" DEL /Q /F "%DDTmpVBS%"
ECHO dt = DateAdd("d",-%MinusDay%,date) >> "%YYYYTmpVBS%"
ECHO yyyy = Year(dt)                    >> "%YYYYTmpVBS%"
ECHO WScript.Echo yyyy                  >> "%YYYYTmpVBS%"
ECHO dt = DateAdd("d",-%MinusDay%,date) >> "%MMTmpVBS%"
ECHO mm = Right("0" ^& Month(dt),2)     >> "%MMTmpVBS%"
ECHO WScript.Echo mm                    >> "%MMTmpVBS%"
ECHO dt = DateAdd("d",-%MinusDay%,date) >> "%DDTmpVBS%"
ECHO dd = Right("0" ^& Day(dt),2)       >> "%DDTmpVBS%"
ECHO WScript.Echo dd                    >> "%DDTmpVBS%"
GOTO EOF

Дополнительные ресурсы

1 голос
/ 11 июня 2012

Попробуйте использовать этот код другими словами. Вы можете использовать в качестве подпрограммы сценария или использовать ее с функциями CALL и параметров для возврата к исходному пакетному файлу:

:: Pass 1st parameter as number of days (whole numbers) to subtract from current day in   date
:: This script is able to subtract days to any date of the current date
:: This script will check for leap years, etc. as well
@echo on
::for /f "tokens=2-4 delims=/ " %%A in ("%date%") do set "mm=%%A" do & set "dd=%%B" do & set "yyyy=%%C"
set "mm=%date:~4,2%" & set "dd=%date:~7,2%" & set "yyyy=%date:~10,4%"
set CurDate=%mm%/%dd%/%yyyy%
set dayCnt=%1
if "%dayCnt%"=="" set dayCnt=1
:: Substract your days here
set /A dd=1%dd% - 100 - %dayCnt%
set /A mm=1%mm% - 100
:CHKDAY
if /I %dd% GTR 0 goto DONE
set /A mm=%mm% - 1
if /I %mm% GTR 0 goto ADJUSTDAY
set /A mm=12
set /A yyyy=%yyyy% - 1
:ADJUSTDAY
if %mm%==1 goto SET31
if %mm%==2 goto LEAPCHK
if %mm%==3 goto SET31
if %mm%==4 goto SET30
if %mm%==5 goto SET31
if %mm%==6 goto SET30
if %mm%==7 goto SET31
if %mm%==8 goto SET31
if %mm%==9 goto SET30
if %mm%==10 goto SET31
if %mm%==11 goto SET30
:: ** Month 12 falls through
:SET31
set /A dd=31 + %dd%
goto CHKDAY
:SET30
set /A dd=30 + %dd%
goto CHKDAY
:LEAPCHK
set /A tt=%yyyy% %% 4
if not %tt%==0 goto SET28
set /A tt=%yyyy% %% 100
if not %tt%==0 goto SET29
set /A tt=%yyyy% %% 400
if %tt%==0 goto SET29
:SET28
set /A dd=28 + %dd%
goto CHKDAY
:SET29
set /A dd=29 + %dd%
goto CHKDAY
:DONE
if /I %mm% LSS 10 set mm=0%mm%
if /I %dd% LSS 10 set dd=0%dd%
echo Date %dayCnt% day(s) before %CurDate% is %mm%/%dd%/%yyyy%
SET DirDate=%mm%/%dd%/%yyyy%
:: The %2 parameter is passed from the calling script as the full path and name of the          file to call back
:: %2 equals %~fnx0
:: The dirdate variable is passed as parameter %1 back to the calling script
Call %2 %dirdate%
GOTO EOF

Я собираюсь искать VB или что-то более эффективное, что я все еще могу включить или вызвать из партии для динамического вычисления дат.

1 голос
/ 07 мая 2012

Мне нужно было что-то, что вычтет дни из текущей даты при проверке високосных лет и т. Д., И это сработало замечательно.

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

Вот примеры:

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

IF "%1"=="" goto modifydate

:modifydate
SET subtractdays=5
SET ModDateScript=\\servershare\path\Called_Scripts\ModDate.cmd
CALL "%ModDateScript%" %subtractdays% "%~fnx0"

Скрипт, который вычислит и передаст параметр% moddate% исходному вызывающему скрипту, который будет установлен как переменная для его соответствующей обработки. Вы просто поместите это в конец скрипта, который вы вызываете, чтобы изменить / вычесть дни из текущей даты (ModDate.cmd).

SET moddate=%mm%/%dd%/%yyyy%
Call %2 %moddate%
GOTO EOF

Мне удалось протестировать и определить, что эти строки из исходного сценария опубликованы:

set yyyy=
set $tok=1-3
for /f "tokens=1 delims=.:/-, " %%u in ('date /t') do set $d1=%%u
if "%$d1:~0,1%" GTR "9" set $tok=2-4
for /f "tokens=%$tok% delims=.:/-, " %%u in ('date /t') do (
for /f "skip=1 tokens=2-4 delims=/-,()." %%x in ('echo.^|date') do (
    set %%x=%%u
    set %%y=%%v
    set %%z=%%w
    set $d1=
    set $tok=))
if "%yyyy%"=="" set yyyy=%yy%
if /I %yyyy% LSS 100 set /A yyyy=2000 + 1%yyyy% - 100

Может быть заменено только одной строкой, и она работает так же хорошо:

FOR /F "tokens=2-4 delims=/ " %%A IN ("%date%") DO SET "mm=%%A" DO (& SET "dd=%%B") DO (& SET "yyyy=%%C")

Пожалуйста, объясните, что эти строки (те, которые я поменял на одну строку с утверждениями) в любом случае делают, потому что я не могу быстро определить разницу при тестировании. Я вычитал еще в 19 веке, и это показалось мне точным. Я подумал, что, возможно, это помогло справиться с расчетами, когда модифицированный год будет меньше 2000 - но я этого не видел, если не упустил что-то.

В противном случае этот один сценарий можно легко вызвать и передать обратно% mm% /% dd% /% yyyy% в качестве параметра для нескольких сценариев, которые нуждаются в собственных вычислениях. Отличное и очень эффективное пакетное решение. Я могу передать аргумент как% 1,% 2,% 3 и т. Д. И все еще использовать setlocal в этом сценарии для текущей даты - просто сделайте переменную что-то вроде moddate =% 1 и т. Д.

Наконец, я призываю любого специалиста по пакетным сценариям еще больше оптимизировать этот сценарий и выкладываю результаты тестов для пакетных пользователей.

Спасибо, P

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...