标题: [日期时间] 批处理获取指定天数之前的日期(2016-06-13更新) [打印本页]
作者: Batcher 时间: 2009-2-14 00:38 标题: 批处理获取指定天数之前的日期(2016-06-13更新)
日期计算是不能直接进行减法(或加法)的,因为涉及到跨月、跨年、闰年等情况。
1582年以来公历的置闰规则:
普通闰年:公历年份是4的倍数,且不是100的倍数的,为闰年(如2004年、2020年等就是闰年)。
世纪闰年:公历年份是整百数的,必须是400的倍数才是闰年(如1900年不是闰年,2000年是闰年)。
1582年以前的惯例:
四年一闰;如果公元A年的A(正数)能被4整除,那么它就是闰年;如果公元前B年的B(正数)除以4余1,那么它也是闰年。
【方案一】BAT调用PowerShell- @echo off
- REM Win7/Win10/Win11等系统自带PowerShell
-
- REM 指定天数
- set DaysAgo=1
- for /f %%i in ('powershell -c "Get-Date (Get-Date).AddDays(-%DaysAgo%) -uformat "%%Y-%%m-%%d""') do (
- set "DstDate=%%i"
- )
- echo,%DstDate%
- pause
复制代码
【方案二】BAT调用VBS- @echo off
- REM 指定天数
- set DaysAgo=1
- >"%temp%\MyDate.vbs" echo LastDate=date()-%DaysAgo%
- >>"%temp%\MyDate.vbs" echo FmtDate=right(year(LastDate),4) ^& right("0" ^& month(LastDate),2) ^& right("0" ^& day(LastDate),2)
- >>"%temp%\MyDate.vbs" echo wscript.echo FmtDate
- for /f %%a in ('cscript /nologo "%temp%\MyDate.vbs"') do (
- set DstDate=%%a
- )
- set DstDate=%DstDate:~0,4%-%DstDate:~4,2%-%DstDate:~6,2%
- echo %DaysAgo%天之前的日期是:%DstDate%
- pause
复制代码
【方案三】BAT调用gawk- @echo off
- REM 本例需要命令行工具gawk.exe的支持,下载地址:
- REM http://bcn.bathome.net/s/tool/index.html?key=gawk
-
- REM 指定天数
- set DaysAgo=1
- set ThisDate=2014-02-20
- echo %ThisDate% | gawk -F "-" -v days=%DaysAgo% "{$1=$1;print strftime(\"%%F\",mktime($0\" 0 0 0\")-86400*days)}"
- pause
复制代码
【方案四】BAT调用日期时间函数- @echo off
- REM 此类方法的关键在于如何获取标准化的当前系统日期
- REM 请参考http://bbs.bathome.cn/thread-3328-1-1.html
- REM 然后调用 Ritchie Lawrence 的日期函数,便可扩展出很多种方法,此处不再赘述。
- REM 日期转换的核心算法请参考http://bbs.bathome.cn/thread-3056-1-1.html
-
- REM 指定天数
- set DaysAgo=1
- for /f "skip=4 delims= " %%a in ('reg query "HKEY_CURRENT_USER\Control Panel\International" /v sShortDate') do (
- set RegDateOld=%%a
- )
- set RegDateOld=%RegDateOld:~-8%
- reg add "HKEY_CURRENT_USER\Control Panel\International" /v sShortDate /t REG_SZ /d yyyy-M-d /f>nul
- call :DateToDays %date:~0,4% %date:~5,2% %date:~8,2% PassDays
- reg add "HKEY_CURRENT_USER\Control Panel\International" /v sShortDate /t REG_SZ /d %RegDateOld% /f>nul
- set /a PassDays-=%DaysAgo%
- call :DaysToDate %PassDays% DstYear DstMonth DstDay
- set DstDate=%DstYear%-%DstMonth%-%DstDay%
- echo %DaysAgo%天之前的日期是:%DstDate%
- pause
- goto :eof
-
- :DateToDays %yy% %mm% %dd% days
- setlocal ENABLEEXTENSIONS
- set yy=%1&set mm=%2&set dd=%3
- if 1%yy% LSS 200 if 1%yy% LSS 170 (set yy=20%yy%) else (set yy=19%yy%)
- set /a dd=100%dd%%%100,mm=100%mm%%%100
- set /a z=14-mm,z/=12,y=yy+4800-z,m=mm+12*z-3,j=153*m+2
- set /a j=j/5+dd+y*365+y/4-y/100+y/400-2472633
- endlocal&set %4=%j%&goto :EOF
-
- :DaysToDate %days% yy mm dd
- setlocal ENABLEEXTENSIONS
- set /a a=%1+2472632,b=4*a+3,b/=146097,c=-b*146097,c/=4,c+=a
- set /a d=4*c+3,d/=1461,e=-1461*d,e/=4,e+=c,m=5*e+2,m/=153,dd=153*m+2,dd/=5
- set /a dd=-dd+e+1,mm=-m/10,mm*=12,mm+=m+3,yy=b*100+d-4800+m/10
- (if %mm% LSS 10 set mm=0%mm%)&(if %dd% LSS 10 set dd=0%dd%)
- endlocal&set %2=%yy%&set %3=%mm%&set %4=%dd%&goto :EOF
复制代码
作者: defanive 时间: 2009-2-14 01:16
所以还是函数库强大。。。
作者: 秋风·飞扬 时间: 2012-5-2 11:58
不行,我水平太差了,要看懂这些有点困难,先去学习别的了,过阵子再来看
作者: yu2n 时间: 2014-12-11 09:26
本帖最后由 yu2n 于 2014-12-11 10:11 编辑
PK第一个方案BAT+VBS版本……- @echo off
- call :_DateAdd 2014.12.30 -5
- echo, 2014.12.30 5天前是 %dt%
- call :_DateAdd 2014-12-30 +5
- echo, 2014-12-30 5天后是 %dt%
- call :_DateAdd 2014/12/30 +500
- echo, 2014/12/30 500天后是 %dt%
- pause
-
- ' 返回日期(修正系统日期分隔符不同的问题) By Yu2n 2014.12.10
- ' 参数1日期,参数2为加减天数
- Goto :Eof
- :_DateAdd
- (echo, If IsDate^("%~1"^) Then
- echo, dt=CDate^("%~1"^)
- echo, Else
- echo, For Each s1 In Split^(". - /"^)
- echo, For Each s2 In Split^(". - /"^)
- echo, If IsDate^(Replace^("%~1",s1,s2^)^) Then dt=CDate^(Replace^("%~1",s1,s2^)^)
- echo, Next
- echo, Next
- echo, End If
- echo, dt = DateAdd^("d", Eval^("%~2"^), dt^)
- echo, WScript.Echo Right^(Year^(dt^),4^) ^& "-" ^& Right^("0" ^& Month^(dt^),2^) ^& "-" ^& Right^("0" ^& Day^(dt^),2^))>"%tmp%\_DateAdd.vbs"
- for /f "delims=" %%i in ('CScript //NoLogo "%tmp%\_DateAdd.vbs"') do set dt=%%i
- goto :Eof
复制代码
结果:- 2014.12.30 5天前是 2014-12-25
- 2014-12-30 5天后是 2015-01-04
- 2014/12/30 500天后是 2016-05-13
- 请按任意键继续. . .
复制代码
另外,再发一个VBS版……- d1 = "2014.03.01"
- nAdd = "-3"
- d2 = DateAdd("d", Eval(nAdd), GetDate(d1))
- d2 = Right(Year(d2),4) & "-" & Right("0" & Month(d2),2) & "-" & Right("0" & Day(d2),2)
- WScript.Echo d2
-
- ' 返回日期(修正系统日期分隔符不同的问题) By Yu2n 2014.12.10
- Function GetDate(ByVal strDate)
- If IsDate(strDate) Then GetDate = CDate(strDate) : Exit Function
- Dim s1, s2
- For Each s1 In Split(". - /")
- For Each s2 In Split(". - /")
- If IsDate(Replace(strDate,s1,s2)) Then GetDate=CDate(Replace(strDate,s1,s2)) : Exit Function
- Next
- Next
- End Function
复制代码
欢迎光临 批处理之家 (http://bbs.bathome.net/) |
Powered by Discuz! 7.2 |