Board logo

标题: 在批处理中如何关闭由批处理打开的程序? [打印本页]

作者: NaturalJ0     时间: 2006-11-15 03:44    标题: 在批处理中如何关闭由批处理打开的程序?

举例:
在批处理中打开某记事本文件。
然后在批处理中要关闭它。

问题如下:
如果用 taskkill 的 /IM 关闭,会发现其它记事本程序的名称也全都是 notepad.exe ,达不到关闭指定的某一个的效果。
如果用 taskkill 的 /PID 关闭,又如何得知所需关闭的进程的 PID 号呢?

或者有比 taskkill 更方便的命令?

前些天就想到这个问题,到现在也没想出好办法,所以今天来听听大家的高见。
作者: electronixtar     时间: 2006-11-15 04:46
我猜,start "test" notepad.exe 是不是给进程一个名字test呢?然后用 taskkill 杀这个名字?
作者: zh159     时间: 2006-11-15 05:00
start "test" notepad.exe显示的还是notepad.exe进程

用 VBS 激活“文件标题 - 记事本”,再 VBS 发送Alt+F4(要先保存的先发送Ctrl+s)
作者: lxmxn     时间: 2006-11-15 05:11

  electronixtar 可能没有试吧?

  这个方法不行~

作者: ccwan     时间: 2006-11-15 05:13
我试了一下,似乎不会,进程里名称也全都是 notepad.exe ,我又试了一下,依次打开几个bat文件,进程里都只显示cmd.exe,关注中……
作者: zh159     时间: 2006-11-15 05:24
简单的BAT转VBS
@echo off
>%Temp%\Temp.vbs echo set WshShell = WScript.CreateObject("WScript.Shell")
>>%Temp%\Temp.vbs echo WScript.Sleep 500
>>%Temp%\Temp.vbs echo WshShell.AppActivate "自己修改文件标题 - 记事本"
>>%Temp%\Temp.vbs echo WScript.Sleep 500
>>%Temp%\Temp.vbs echo WshShell.SendKeys "^s"
>>%Temp%\Temp.vbs echo WScript.Sleep 500
>>%Temp%\Temp.vbs echo WshShell.SendKeys "%%{F4}"
%Temp%\Temp.vbs


  Quote:
>>%Temp%\Temp.vbs echo WshShell.SendKeys "^s"

保存文件

  Quote:
>>%Temp%\Temp.vbs echo WshShell.AppActivate "自己修改文件标题 - 记事本"

“自己修改文件标题”一般是文件名(如果文件夹选项打开显示文件扩展名则要加入扩展名)
如果是多个同文件标题的好像是随机的-_-||
作者: pengfei     时间: 2006-11-15 05:52
1.  把批处理运行前记事本程序的PID值记录下来.

2.  打开记事本后, 将前后的PID值进行比对, 终止不匹配的PID所对应的程序.
@echo off

setlocal enabledelayedexpansion

for /f "tokens=1,2" %%i in ('tasklist /nh^|findstr /i /c:notepad.exe') do (

    set pid=!pid! %%j

)

::  上面的代码加到批处理的头部.  

start notepad.exe

::  下面的代码放到你想要终止进程的地方.

if "%pid%"=="" set pid=lack

for /f "tokens=1,2" %%i in ("%pid%") do (

    for /f "tokens=1,2" %%a in ('tasklist /nh^|findstr /i /c:notepad.exe') do (

        if not "%%i"=="%%b"  taskkill /pid %%b >nul

    )

)
[ Last edited by pengfei on 2006-11-15 at 10:41 PM ]
作者: namejm     时间: 2006-11-15 06:04
  其实,taskkill 命令提供了一个匹配窗口标题的参数,它就是 /fi 过滤器里的 WINDOWTITLE 选项,请看:
TASKKILL [/S system [/U username [/P [password]]]]
         { [/FI filter] [/PID processid | /IM imagename] } [/F] [/T]
……
    /FI   filter           指定筛选进或筛选出查询的
                           的任务。
……
筛选器:
    筛选器名      有效运算符                有效值
    -----------   ---------------           --------------
……
    WINDOWTITLE     eq, ne                    窗口标题
……
  比如,我要关闭窗口标题为 taskkill.txt - 记事本 的文本文件,此时,用 taskkill /fi "windowtitle eq taskkill*" 就可以了。不过有点奇怪的是,这个选项中必须使用通配符*,否则,就不能关闭。
作者: 3742668     时间: 2006-11-15 08:25
Re namejm:
    匹配窗口需要加*的情况其实并不奇怪。因为windows窗口的标题本来就并非纯粹当前实例指定的标题,而是当前实例标题+窗口类的标题。例如我们打开联盟的IE窗口,就可以看到标题栏除了当前主题的标题外后面还加上了版块标题,联盟页标题以及系统自动加上的Microsoft Internet Explorer,系统加上的部分就是默认此类窗口标题。可以通过修改注册表来更改,记得以前是保存在hklm\software\microsoft\internetexplorer\main下一个包含字符title的键中,不过sp2中似乎更改了位置,但也可以通过gpedit.msc来修改。
    在命令行中,获得完整的标题可以用tasklist /v来查看,查看前建议用mode设定加大缓冲区的宽度,否则会自动换行难以查看。
Re NaturalJ0:
    结束指定进程最精确的莫过于通过pid来指定了,通过两次tasklist列表并比较差别可以得到启动进程的pid值,不过比较繁琐,还好可以通过wmic来简单实现:
@echo off
    for /f "tokens=2 delims==;" %%i in ('wmic process call create notepad ^| findstr /i "ProcessId"') do @set /a x=%%i
    echo 由本脚本启动的notepad进程的pid为:%x%
pause
另外,wmic process可以获得进程的ParentProcessId(父进程pid)和ProcessId(pid),通过ParentProcessId也可以获得父进程的进程名。
作者: pengfei     时间: 2006-11-15 09:48
namejm版主用taskkill /fi 结束指定进程的办法简单省事.

而7楼通过比对两次PID值来终止后来开启的进程的方法. 3742668版主居然用wmic process一个语句就搞定了.

看来有时间得好好学习一下DOS命令了.
作者: lxmxn     时间: 2006-11-15 09:52


  Quote:
查看前建议用mode设定加大缓冲区的宽度,否则会自动换行难以查看。


  用tasklist /v /fo list查看标题就比较清晰了。

作者: NaturalJ0     时间: 2006-11-15 22:26
大家的方法真多啊。

如果不指定 PID ,似乎还是没法避免重名。 IM 或 TITLE 都有可能会相同。
IM 相同:比如运行两个记事本
TITLE 相同:比如用记事本打开两个同名文件(也可以是同一文件打开两次)

如果使用运行前比较的方法,似乎也不是很完美。
比如批处理打开记事本后定时两分钟后自动关闭它,而在这两分钟内又有多个记事本被打开。

3742668 版主的方法我试着换了其它程序,把记事本换成了 curl (上传文件),换成了播放器(播放视频)。但我都没试成功。(我不懂 WMIC ,可能是我不会用,用错了。不知应该如何使用。)

看了大家的讨论还真是学了不少新东西。非常感谢。^_^
作者: pengfei     时间: 2006-11-15 22:38


  Quote:
Originally posted by NaturalJ0 at 2006-11-15 22:26:
如果使用运行前比较的方法,似乎也不是很完美。
比如批处理打开记事本后定时两分钟后自动关闭它,而在这两分钟内又有多个记事本被打开。

代码稍微修改一下就可以符合你的要求了, 灵活...
@echo off

setlocal enabledelayedexpansion

for /f "tokens=1,2" %%i in ('tasklist /nh^|findstr /i /c:notepad.exe') do (

    set pid=!pid! %%j

)

::  上面的代码加到打开记事本的上行.  

start notepad.exe

::  中间的代码放到你刚打开事本的下行.

if "%pid%"=="" set pid=lack

for /f "tokens=1,2" %%i in ("%pid%") do (

    for /f "tokens=1,2" %%a in ('tasklist /nh^|findstr /i /c:notepad.exe') do (

        if not "%%i"=="%%b"  set kill=!kill!/pid %%b

    )

)

::  下面的代码为终止记录下来的记事本进程.

taskkill %kill%
[ Last edited by pengfei on 2006-11-15 at 10:48 PM ]
作者: NaturalJ0     时间: 2006-11-15 23:41
Re: pengfei

  Quote:
@echo off
cd.>不应该关闭的1.txt
cd.>不应该关闭的2.txt
cd.>不应该关闭的3.txt
cd.>不应该关闭的4.txt
cd.>不应该关闭的5.txt
cd.>应该关闭的.txt

start notepad.exe 不应该关闭的1.txt
start notepad.exe 不应该关闭的2.txt
start notepad.exe 不应该关闭的3.txt
start notepad.exe 不应该关闭的4.txt
start notepad.exe 不应该关闭的5.txt

setlocal enabledelayedexpansion

for /f "tokens=1,2" %%i in ('tasklist /nh^|findstr /i /c:notepad.exe') do (

    set pid=!pid! %%j

)

::  上面的代码加到打开记事本的上行.  

start notepad.exe 应该关闭的.txt

::  中间的代码放到你刚打开事本的下行.

if "%pid%"=="" set pid=lack

for /f "tokens=1,2" %%i in ("%pid%") do (

    for /f "tokens=1,2" %%a in ('tasklist /nh^|findstr /i /c:notepad.exe') do (

        if not "%%i"=="%%b"  set kill=!kill!/pid %%b

    )

)

::  下面的代码为终止记录下来的记事本进程.
echo 5 seconds to close
ping 1 -n 1 -w 5000>nul


taskkill %kill%

echo Is everything OK?
echo Now any key to delete the temp files.
pause>nul
taskkill /f /im "notepad.exe"
del 不应该关闭的?.txt
del 应该关闭的.txt

看了你的代码,我没理解。
我在你的代码中直接加了些(红色部分是我加的,黑色部分是你写的)。
你也可以试试效果。

[ Last edited by NaturalJ0 on 2006-11-15 at 11:43 PM ]
作者: pengfei     时间: 2006-11-15 23:58
我想这样你更容易看懂, 主要是灵活使用代码记录下批处理中打开记事本的PID.
@echo off
cd.>不应该关闭的1.txt
cd.>不应该关闭的2.txt
cd.>应该关闭的1.txt
cd.>应该关闭的2.txt

start 不应该关闭的1.txt

setlocal enabledelayedexpansion

for /f "tokens=1,2" %%i in ('tasklist /nh^|findstr /i /c:notepad.exe') do (

    set pid=!pid! %%j

)

::  上面的代码加到打开记事本的上行.  

start 应该关闭的1.txt

start 应该关闭的2.txt

::  中间的代码放到你刚打开事本的下行.

if "%pid%"=="" set pid=lack

for /f "tokens=1,2" %%i in ("%pid%") do (

    for /f "tokens=1,2" %%a in ('tasklist /nh^|findstr /i /c:notepad.exe') do (

        if not "%%i"=="%%b"  set kill=!kill!/pid %%b

    )

)

start notepad.exe 不应该关闭的2.txt

::  下面的代码为终止记录下来的记事本进程.

echo 现在你可以在WINDOWS下打开任意记事本, 按任意键关闭记录的进程.

pause >nul

taskkill %kill% >nul

echo 进程成功终止!

pause>nul

作者: vkill     时间: 2006-11-16 03:49
wmic 也是根据进程名结束的
作者: 3742668     时间: 2006-11-16 14:02


  Quote:
3742668 版主的方法我试着换了其它程序,把记事本换成了 curl (上传文件),换成了播放器(播放视频)。但我都没试成功。(我不懂 WMIC ,可能是我不会用,用错了。不知应该如何使用。)


@echo off
    set /p str=输入程序名(空格加引号):
    for /f "tokens=2 delims==;" %%i in ('wmic process call create %str% ^| findstr /i "ProcessId"') do @set /a x=%%i
    echo 由本脚本启动的%str%进程的pid为:%x%
pause


  Quote:
wmic 也是根据进程名结束的

不仅wmic可以通过多种属性识别进程,单是taskkill就可以通过多种属性结束指定进程,就连ntsd在XP中都可以分别通过pid和imagename来结束指定进程。

ntsd -c q -p %pid%
ntsd -c -q -pn %imagename%
taskkill /fi "pid eq/ne/gt/lt/ge/le %pid%"
taskkill /im %imagename%
taskkill /pid %pid%
wmic process where "processid='%pid%'" call Terminate
wmic process where "name='%imagename%'" call Terminate
wmic path  win32_process.name="%imagename%" call Terminate
wmic path win32_process.processid="%pid%" call Terminate
...

作者: voiL     时间: 2006-11-18 12:01


  Quote:
Originally posted by 3742668 at 2006-11-16 14:02:

[code]@echo off
    set /p str=输入程序名(空格加引号):
    for /f "tokens=2 delims==;" %%i in ('wmic process call create %str% ^| findstr /i "ProcessId"') do @set  ...

好家伙,之前已经领教过3742668版主的wmic了,想不到这一次又是wmic,看来得好好学学wmic才行.
作者: pengfei     时间: 2006-11-18 12:37
3742668版主的代码是否可以修改成打开文件, 如果只是打开程序有点美中不足.
作者: 3742668     时间: 2006-11-20 13:59


  Quote:
3742668版主的代码是否可以修改成打开文件, 如果只是打开程序有点美中不足.

当然可以,只要把create后面的命令用“cmd /c start "的格式来替换就行了,然后在结束进程的时候不能用processid=%pid%来结束,而是用上面提到过的parentprocessid来解决,因为cmd /c运行后会自动退出,而当时获得的pid为cmd的pid,是被启动的进程的父进程。
作者: NaturalJ0     时间: 2006-11-20 21:37
发个图
附件 1: cmd.JPG (2006-11-20 21:37, 41.69 K, 下载附件所需积分 1点 ,下载次数: 19)



作者: vlq5299     时间: 2006-12-5 03:21
有点麻烦,有没有简单的?