我有一个在Windows上创建进程的函数。
pub fn create_process(url: String) {
thread::spawn(move || {
let _child = process::Command::new("cmd.exe")
.arg("/C")
.arg("ping")
.arg(&url)
.arg("-t")
.spawn()
.expect("Couldn't run 'ping'");
});
}
我有一个函数想要终止(杀死)“ create_process()”创建的进程:
pub fn stop() {
// ?????
}
如何访问在_child
函数中创建的create_process
以终止其过程?还有其他方法可以终止该进程吗?
答案 0 :(得分:5)
TL; DR:使用kill方法。例如:
use std::process;
use std::thread;
use std::time::Duration;
fn main() {
let mut child = process::Command::new("ping")
.arg("8.8.8.8")
.arg("-t")
.spawn()
.expect("Couldn't run 'ping'");
thread::sleep(Duration::from_secs(5));
child.kill().expect("!kill");
}
请注意,由于生成的进程已经与父进程平行,因此您不需要单独的线程。
在您的问题中,有一个代码示例使用“ cmd / C”运行“ ping”。这产生的不是一个过程,而是两个过程:“ cmd”过程和“ ping”过程。终止child
将终止“ cmd”进程,但可能会使“ ping”进程继续运行。
使用“ cmd / C”也是危险的,允许使用command injection。
如何访问在
_child
函数中创建的create_process
以终止其过程?
_child
is Send
,这意味着您可以在线程之间发送它。多个相应的堆栈溢出问题可能已经涵盖了跨线程发送数据的细节。
还有其他方法可以终止该进程吗?
您可以使用本机平台API。例如:
[dependencies]
gstuff = "0.5.2"
winapi = {version = "0.3.6", features = ["psapi", "shellapi"]}
#[macro_use]
extern crate gstuff;
use std::process;
use std::ptr::null_mut;
use std::thread;
use std::time::Duration;
use winapi::shared::minwindef::DWORD;
use winapi::shared::ntdef::HANDLE;
use winapi::um::processthreadsapi::{OpenProcess, TerminateProcess};
use winapi::um::winnt::{PROCESS_QUERY_INFORMATION, PROCESS_TERMINATE};
struct Process(HANDLE);
impl Process {
fn open(pid: DWORD) -> Result<Process, String> {
// https://msdn.microsoft.com/en-us/library/windows/desktop/ms684320%28v=vs.85%29.aspx
let pc = unsafe { OpenProcess(PROCESS_QUERY_INFORMATION | PROCESS_TERMINATE, 0, pid) };
if pc == null_mut() {
return ERR!("!OpenProcess");
}
Ok(Process(pc))
}
fn kill(self) -> Result<(), String> {
unsafe { TerminateProcess(self.0, 1) };
Ok(())
}
}
impl Drop for Process {
fn drop(&mut self) {
unsafe { winapi::um::handleapi::CloseHandle(self.0) };
}
}
fn main() {
let child = process::Command::new("ping")
.arg("8.8.8.8")
.arg("-t")
.spawn()
.expect("Couldn't run 'ping'");
let pid = child.id();
let pc = Process::open(pid as DWORD).expect("!open");
println!("Process {} opened.", pid);
thread::sleep(Duration::from_secs(5));
pc.kill().expect("!kill");
println!("Process {} killed.", pid);
}