我创建了一个 C 程序,它将运行一些system()
命令并将输出保存到 C 驱动器特定文件夹中的.txt
文件中(程序将在具有启动权限的域客户端上运行),从那里我将过滤数据并将其显示在输出控制台屏幕上。
一切正常,但我无法为那些在其他驱动器中安装了操作系统(例如 D:
、E:
等)的 PC 设计它,因为它们的C:
驱动器中没有这个特定的文件夹。由于组策略,我无法在其他任何地方写入临时.txt
文件。
有没有办法将此数据直接流式传输到任何数组变量中?我浏览了popen()
函数,但它需要定义一个非常大的不可预测大小数组(因为system()
命令的输出可能非常大)。例如,将HKLMSOFTWAREMicrosoftWindowsCurrentVersionuninstall
的注册表项导出到.txt
文件中。它的大小可能高达 50KB 或更大。
reg export HKLMSOFTWAREMicrosoftWindowsCurrentVersionuninstall C:\..(somewhere)....\reg_output.txt
还有一些具有大输出的类似命令。我还不知道 Win API 编程,因此我正在使用系统命令。有没有一个简单的替代方法来写.txt
?
@Megharaj!正如你用过的
char line[100];
fp = popen("ifconfig eth0", "r");
fgets(line, 100, fp);
我将不得不使用
char reg_output[100000];
fp=popen("reg export HKLMSOFTWAREMicrosoftWindowsCurrentVersionuninstall ????","r");
fgets(line,100000,fp);
在哪里导出注册表值?(由于用于导出此命令的DOS命令需要将其写入文件),与我使用的以下代码相比。
分配 100000 的空间不确定它是否容易出错。分配过高的值也会影响启动时的内存(我不确定,但猜是这样)。
在使用文件处理时,我这样做是:
char line[5000]; FILE* fp_reg; int ch,n=0;
system("reg export HKLMSOFTWAREMicrosoftWindowsCurrentVersionuninstall c:\registries.txt");
fp_reg=fopen("c:\registries.txt","r");
while((ch=fgetc(fp_reg))!=EOF)
{
if(isalnum(ch) || ch=='n') //allow only alpha numeric & 'n' to pass to array
{ line[n]=ch;
if(ch=='n')
{ filter_for_software(line,n); //Send the array for filtering function
n=0; // Reset the array 'line'
}
n++;
}
}
(我没有从源代码复制代码,因为我坐在另一台 PC 上。但代码几乎与此类似。如果您复制此代码并运行,可能会出现一些错误。这样,我从文件中获取每一行并发送它进行过滤以获取"已安装软件的名称"。如何使用"popen()"来做到这一点,以便我可以拿起一行并将其扔到函数上。请为此编写一些代码。
将输出存储在数组中,则必须分配适当大小的数组。这是没有办法的。
但是,如果您使用 popen
.您将获得一个FILE *
句柄,只需读取小部分的输出并在旅途中进行处理。
从那里我将过滤数据并将其显示在输出控制台屏幕上。
你是怎么做到的?用外壳还是在 C 中?如果在 C 中,则读取 system
ed 命令的输出,就像读取 txt 文件一样。唯一的区别是您需要用 pclose
而不是 fclose
关闭它,因此使用文本文件这样做只有优点。
如果在shell中,那么你可以启动你使用的程序,另一个调用popen
,这次是在一个"w"
(写)方向上,并将一个管道的输出作为输入写入另一个管道。但是,在这种情况下,您可以首先使用匿名管道调用 shell。;-)
编辑:
你自己的答案清楚地表明,你的要求与你认为的完全不同。没有办法通过管道读回reg export
的输出,如果它坚持写入文件,popen
。
您可以尝试将文件写入临时文件夹;您应该被允许在那里创建文件,否则Windows将无法正常工作。只需指定类似"%TEMP%\reg.out"的内容作为文件,然后从那里读回数据。
如果这不起作用,那么您就不走运了reg export
.但是您可以使用一些Windows API函数直接查询注册表。这是一个起点。
或者,您可能希望研究使用PowerShell的可能性。您可能会对此问题感兴趣。
在 linux 中使用系统命令中的值,我使用 popen,我只是举一个我很久以前编写的代码示例,通过系统命令"ifconfig eth0"获取 pc 的 ip 地址到字符串/a 文件中。 查看示例
void get_my_ip(char *ip_mac_address)
{
FILE *fp,*output;
char *start=NULL;
char *end=NULL;
char line[100];
output=fopen("my_ip_address.txt", "w");
if(output == NULL) {
printf("error creating outputfilen");
return -1;
}
printf("program to self query the ip addressn");
fp = popen("ifconfig eth0", "r");
fgets(line, 100, fp);
start=strstr(line, CHECK_STRING_MAC);
start = start + 7;
fwrite(start, 1, 17, output); start = NULL;
fgets(line, 100, fp);
start=strstr(line, CHECK_STRING_IP);
start = start + 10;
fwrite(start, 1, 14, output);
fclose(output);
pclose(fp);
if( access("my_ip_address.txt", F_OK ) != -1 ) {
printf("found file having ip addressn");
output=fopen("my_ip_address.txt", "r");
fgets(ip_mac_address, 32, output);
}
else
printf("unabe to find file with ip addressn");
fclose(output);
printf("my ip and mac address adress is %s n",ip_mac_address);
}
您可以使用Windows中的某些API创建一个临时文件,并将数据存储在其中。
http://msdn.microsoft.com/en-us/library/windows/desktop/aa363875(v=vs.85).aspx
链接中有一个C++示例,您应该能够将其适应您的情况。
因此,我准备了代码,以将DOS命令的输出流式传输到读取模式下的文件指针,并逐行检查:
char array_sys[200]; FILE *fp_sys; int ch,n=0;
fp=popen("systeminfo","r"); //runs a DOS command & read it with a file pointer
if(fp_sys==NULL) printf("can't open filen");
while((ch=fgetc(fp_sys))!=EOF) //READ CHARACTERS FROM FILE POINTER UNTIL FILE ENDS
{
array_sys[n]=ch; //ASSIGN CH TO EACH ELEMENT OF ARRAY
n++; //INCREMENT ELEMENTS OF ARRAY:-arr_sys
if(ch=='n') //IF ELEMENTS MEET A NEW LINE CHARACTER
{
disp_date(array_sys,n); //PASS ARRAY TO FUNCTION:-disp_date
n=0; //RESET THE ARRAY
}
}
现在这就是我在函数中处理数组以获取操作系统名称的方式。
void disp_date(char array_sys[],int ind)
{
char os_name[9]={"OS Name:"};
if(strstr(array_sys,os_name)) //IF 'OS NAME' IS PRESENT IN ARRAY
{
printf("%s",array_sys); //PRINT THE ARRAY
}
}