你可能会疑问:直接使用system()函数不就完了吗?干嘛还写一大堆呢?
事实上,cmd是一个已经搭建好的简易编程语言,如果把它放在远程、爬虫等方面是非常实用的,因为在远程的时候需要很多监视(返回)信息,所以有了这个标题。
从网上一些博客来看,很多都是用popen写的,这里有一个示例:
string cmd(string s)
{
FILE* fp = NULL;
char cmd[512];
sprintf(cmd, s.c_str());
if ((fp = popen(cmd, "r")) != NULL)
{
fgets(cmd, sizeof(cmd), fp);
pclose(fp);
}
return cmd;
}
popen的好处是代码简单好写,适用于简单的命令执行和输出捕获,但是被限制了很多windows数据,无法获取一些精准的信息(比如netstat -ano)
那如果获取windows信息怎么办呢?我们可以很容易想到一种方法:管道
这里简单介绍一下:CreateProcess + 管道(可能有一些不精准,勿喷)
平台:是Windows API的一部分,专门用于Windows操作系统。依赖于Windows的进程管理机制,因此在Windows上使用非常方便,但在其他操作系统上不可用。
灵活性:可以同时捕获标准输出和标准错误输出,可以对子进程进行更精细的控制,比如设置环境变量、指定工作目录、控制子进程的优先级等。
实现复杂度:实现较为复杂,需要手动创建管道、设置进程启动信息、处理句柄等。
代码:(点赞和评论的人最帅了~~~😄)
#include <windows.h>
#include <bits/stdc++.h>
using namespace std;
string RunCommand(const std::string& command) {
// 定义管道的读取端和写入端
HANDLE hReadPipe, hWritePipe;
SECURITY_ATTRIBUTES sa;
// 设置安全属性,允许管道句柄被子进程继承
sa.nLength = sizeof(SECURITY_ATTRIBUTES);
sa.bInheritHandle = TRUE;
sa.lpSecurityDescriptor = NULL;
// 创建管道
if (!CreatePipe(&hReadPipe, &hWritePipe, &sa, 0)) {
return "CreatePipe failed!";
}
// 设置STARTUPINFO结构体,指定标准输出和标准错误输出重定向到管道
STARTUPINFO si;
PROCESS_INFORMATION pi;
ZeroMemory(&si, sizeof(STARTUPINFO));
si.cb = sizeof(STARTUPINFO);
si.hStdError = hWritePipe;
si.hStdOutput = hWritePipe;
si.dwFlags |= STARTF_USESTDHANDLES;
// 创建进程
if (!CreateProcess(
NULL, // 应用程序名称
const_cast<char*>(command.c_str()), // 命令行
NULL, // 进程安全描述符
NULL, // 线程安全描述符
TRUE, // 继承句柄
0, // 创建标志
NULL, // 环境变量
NULL, // 当前目录
&si, // STARTUPINFO
&pi // PROCESS_INFORMATION
)) {
CloseHandle(hReadPipe);
CloseHandle(hWritePipe);
return "CreateProcess failed!";
}
// 关闭写入端,因为我们不需要写入
CloseHandle(hWritePipe);
// 读取管道中的数据
char buffer[4096];
DWORD bytesRead;
std::string output;
while (true) {
if (ReadFile(hReadPipe, buffer, sizeof(buffer), &bytesRead, NULL)) {
if (bytesRead == 0) break;
output.append(buffer, bytesRead);
} else {
break;
}
}
// 关闭读取端
CloseHandle(hReadPipe);
// 等待子进程结束
WaitForSingleObject(pi.hProcess, INFINITE);
// 关闭进程和线程句柄
CloseHandle(pi.hProcess);
CloseHandle(pi.hThread);
return output;
}
int main() {
// 运行CMD命令并捕获输出
cout << RunCommand("ipconfig /all");
return 0;
}
代码说明:
- CreatePipe: 创建一个管道,用于捕获CMD命令的输出。
- STARTUPINFO: 设置子进程的标准输出和标准错误输出重定向到管道。
- CreateProcess: 创建一个子进程来执行CMD命令。
- ReadFile: 从管道中读取CMD命令的输出。
- WaitForSingleObject: 等待子进程结束。
点赞!!!关注!!!评论!!!求求啦!!!😆😆