如何杀死运行时间超过30分钟的Windows进程

Ame*_*mey 11 windows service process batch-file

我基本上需要实现两件事,

  1. 找到在Windows服务器上运行的所有"firefox.exe"进程
  2. 杀死跑步时间超过30分钟的人

我有点零碎,但不知道如何整合所有它使它作为一个Windows服务.

到目前为止我所拥有的 -

1)找到所有正在运行的firefox进程的方法

wmic process get name,creationdate, processid | findstr firefox
Run Code Online (Sandbox Code Playgroud)

2)基于PID杀死进程的方法

taskkill /PID 827
Run Code Online (Sandbox Code Playgroud)

还剩下什么?

  • 计算基于creationdate,PID运行时间超过30分钟
  • 使用该taskkill命令可以按顺序终止符合上述条件的所有PID
  • 设置这是一项服务(这我可以搞清楚)

jim*_*ark 19

很容易想到,"你不能在.bat中做到这一点".我知道这是我的第一反应.问题是你需要日期操作,这不是直接支持的,并且是非平凡的.但随后Ritchie Lawrence开始拯救,完成了编写必要日期功能的所有艰苦工作.

WMI ProcessCreationDate以UTC格式提供.Win32_OperatingSystem LocalDateTime以UTC格式给出了当前时间.我们需要从LocalDataTime中减去最大生命周期(在你的情况下为30分钟)以获得CutOffTime.然后我们可以使用它来过滤Process,最后call terminate(而不是taskkill).而不是findstr,我使用WMI where过滤器(更快).

以下代码似乎有效.由于这是KILLING TASKS,你应该自己测试一下.

注意:if "%%p" GEQ "0"用于过滤wmic结果的"空白"行,该行不是空的但包含换行符.正如我们期待的那样,这似乎是一个简单而有效的测试(尽管可能有更好的方法来解决这个问题).

@echo off
setlocal ENABLEDELAYEDEXPANSION ENABLEEXTENSIONS

set MaxRunningMinutes=30
set ProcessName=firefox.exe

for /f "usebackq skip=1" %%t in (
  `wmic.exe path Win32_OperatingSystem get LocalDateTime`) do (
    if "%%t" GEQ "0" set T=%%t)

rem echo !T!
rem echo !T:~,4!/!T:~4,2!/!T:~6,2! !T:~8,2!:!T:~10,2!:!T:~12,2!
rem echo !T:~15,-4! !T:~-4!

set fsec=!T:~15,-4!
set tzone=!T:~-4!

call :DateToSecs !T:~,4! !T:~4,2! !T:~6,2! !T:~8,2! !T:~10,2! !T:~12,2! UNIX_TIME
rem echo !UNIX_TIME!

set /a CutOffTime=UNIX_TIME-MaxRunningMinutes*60
rem echo !CutOffTime!

call :SecsToDate !CutOffTime! yy mm dd hh nn ss
rem echo !yy!/!mm!/!dd! !hh!:!nn!:!ss!
set UTC=!yy!!mm!!dd!!hh!!nn!!ss!.!fsec!!tzone!
rem echo !UTC!

wmic process where "name='%ProcessName%' AND CreationDate<'%UTC%'" call terminate

rem * Alternate kill method. May be useful if /F flag is needed to
rem * to forcefully terminate the process. (Add the /F flag to
rem * taskill cmd if needed.)

rem for /f "usebackq skip=1" %%p in (
rem   `wmic process where "name='%ProcessName%' AND CreationDate<'%UTC%'" get processid`) do (
rem   if "%%p" GEQ "0" taskkill /PID %%p)

goto :EOF

rem From: http://www.commandline.co.uk/lib/treeview/index.php
:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
:DateToSecs %yy% %mm% %dd% %hh% %nn% %ss% secs
::
:: By:   Ritchie Lawrence, updated 2002-08-13. Version 1.1
::
:: Func: Returns number of seconds elapsed since 1st January 1970 00:00:00
::       for a given calendar date and time of day. For NT4/2000/XP/2003.
::
:: Args: %1 year to convert, 2 or 4 digit (by val)
::       %2 month to convert, 1/01 to 12, leading zero ok (by val)
::       %3 day of month to convert, 1/01 to 31, leading zero ok (by val)
::       %4 hours to convert, 1/01 to 12 for 12hr times (minutes must be
::          suffixed by 'a' or 'p', 0/00 to 23 for 24hr clock (by val)
::       %5 mins to convert, 00-59 only, suffixed by a/p if 12hr (by val)
::       %6 secs to convert, 0-59 or 00-59 (by val)
::       %7 var to receive number of elapsed seconds (by ref)
:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::

setlocal ENABLEEXTENSIONS
set yy=%1&set mm=%2&set dd=%3&set hh=%4&set nn=%5&set ss=%6
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
if 1%hh% LSS 20 set hh=0%hh%
if {%nn:~2,1%} EQU {p} if "%hh%" NEQ "12" set hh=1%hh%&set/a hh-=88
if {%nn:~2,1%} EQU {a} if "%hh%" EQU "12" set hh=00
if {%nn:~2,1%} GEQ {a} set nn=%nn:~0,2%
set /a hh=100%hh%%%100,nn=100%nn%%%100,ss=100%ss%%%100
set /a j=j*86400+hh*3600+nn*60+ss
endlocal&set %7=%j%&goto :EOF
:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::

:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
:SecsToDate %secs% yy mm dd hh nn ss
::
:: By:   Ritchie Lawrence, updated 2002-07-24. Version 1.1
::
:: Func: Returns a calendar date and time of day from the number of
::       elapsed seconds since 1st January 1970 00:00:00. For
::       NT4/2000/XP/2003.
::
:: Args: %1 seconds used to create calendar date and time of day (by val)
::       %2 var to receive year, 4 digits for all typical dates (by ref)
::       %3 var to receive month, 2 digits, 01 to 12 (by ref)
::       %4 var to receive day of month, 2 digits, 01 to 31 (by ref)
::       %5 var to receive hours, 2 digits, 00 to 23 (by ref)
::       %6 var to receive minutes, 2 digits, 00 to 59 (by ref)
::       %7 var to receive seconds, 2 digits, 00 to 59 (by ref)
:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
setlocal ENABLEEXTENSIONS
set /a i=%1,ss=i%%60,i/=60,nn=i%%60,i/=60,hh=i%%24,dd=i/24,i/=24
set /a a=i+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%)
(if %hh% LSS 10 set hh=0%hh%)&(if %nn% LSS 10 set nn=0%nn%)
if %ss% LSS 10 set ss=0%ss%
endlocal&set %7=%ss%&set %6=%nn%&set %5=%hh%&^
set %4=%dd%&set %3=%mm%&set %2=%yy%&goto :EOF
:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
Run Code Online (Sandbox Code Playgroud)

使用Powershell

最初的问题被标记为"批处理文件",所以我首先给出了Windows bat答案.但是我想提到杀死一个运行时间超过30分钟的命名进程与Powershell一样微不足道:

Get-Process firefox | 
    Where StartTime -lt (Get-Date).AddMinutes(-30) | 
    Stop-Process -Force
Run Code Online (Sandbox Code Playgroud)

这是杀死任务所以请小心.如果你不熟悉Powershell这里是一个班轮的细分:

  1. 使用Get-Processcmdlet获取本地计算机上运行的进程.这里firefox传入了进程的名称.如果需要,可以传递名称列表,并且可以使用通配符.
  2. 返回的进程对象通过Get-Process管道传递Where以过滤StartTime属性.该Get-Datecmdlet用于获取当前日期和时间作为一个DateTime类型.调用AddMinutes方法添加-30分钟,返回DateTime30分钟前的代码.的-lt操作者(在技术上,在这种情况下,这是一个开关参数),指定低于操作.
  3. 返回的过滤过程对象通过Where管道传递给Stop-Processcmdlet.该-Force参数用于防止确认提示.

上面我使用了WherePowershell 3中引入的简化语法.如果你需要Powershell 2兼容性(这是在Windows XP上运行的最新版本),那么这个语法是必需的:

Get-Process firefox | 
    Where { $_.StartTime -lt (Get-Date).AddMinutes(-30) } | 
    Stop-Process -Force
Run Code Online (Sandbox Code Playgroud)

这里的花括号包含一个ScriptBlock.$_用于显式引用当前对象.