gpt4 book ai didi

c - Perl 将数据传递给 stdin 并从 stdout 和 stderr 读取

转载 作者:太空宇宙 更新时间:2023-11-04 00:54:28 24 4
gpt4 key购买 nike

这里可能有很多关于相同问题的问题,但目前没有一个对我有用。

我有一个C程序如下:

#include <stdio.h>
#include <sys/stat.h>

long size(FILE *st_in) {
struct stat st;
if (fstat(fileno(st_in), &st) == 0)
return st.st_size;
return -1;
}

int main (){
FILE *file = stdin;
long s1, s2;

s1 = size(file);
printf("Size is %ld\n", s1);
fprintf(stderr, "%d", 1);
return 0;
}

我将其编译为输出 a.out。

我有一个 xml 文件“sample.xml”

<sample>
<tag>Hello</tag>
<tag>Cool</tag>
</sample>

然后我有这个 perl 代码

#!/usr/bin/perl
use warnings;
use IPC::Open2;
open FILE, "sample.xml" or die $!;
my @xmlfile = <FILE>;
my $pid = open2(*Reader, *Writer, "./a.out");
print Writer "@xmlfile";
waitpid($pid, 0);
my @got = <Reader>;
close Writer;
close Reader;
close $pid;
print "Output got is\n";
print @got;

如果我运行通过标准输入传递 simple.xml 的 C 程序,我得到以下信息:

[bharath@xml 18:22:34]$ ./a.out < sample.xml 
Size is 60

当我运行 perl 代码时,我希望大小为 60。但我得到的是 0。

[bharath@xml 18:22:42]$ ./firmware.pl 
Output got is
Size is 0

那我该如何解决呢?我需要从 perl 中的@array 传递 sample.xml。来自 C 程序的 stderr 整数应该存储在一个单独的变量中,而来自 C 程序的标准输出应该存储在 perl 中的另一个单独的变量中。我认为这可能需要使用 open3,但我不知道如何使用。任何工作示例将不胜感激。

最佳答案

更新:关于您的评论,Ilmari Karonen 已经向您解释过,管道没有文件大小,因为它是数据流,程序不知道该流有多大。

你有两个问题:你的 C 程序不工作,而且你的 Perl 程序不工作,因为它进入了死锁状态。你不能同时测试这两件事。把这两个问题分开。例如,首先使用 shell 中的管道尝试您的程序。

cat sample.xml | ./a.out 
Size is 60

最初它不起作用。为了让它工作,我使用了这个修改过的 C 程序:可以通过计算 EOF 之前所有接收到的字符来从流中计算大小。

#include <stdio.h>

int main (){
long size = 0;
int ch;
FILE *file = stdin;
if (!file) {
return 2;
}
while ((ch = getc(file)) != EOF) {
++size;
}
printf("Size is %ld\n", size);
fprintf(stderr, "%d", 1);
return 0;
}

至于你的 Perl 程序,你有一个死锁,因为两个程序都处于等待状态,为了解决它,我稍微改变了指令的顺序:

#!/usr/bin/perl -w

use strict;
use IPC::Open2;

open FILE, "sample.xml" or die $!;
my @xmlfile = <FILE>;

my $pid = open2(*Reader, *Writer, './a.out 2> /dev/null');
print Writer @xmlfile;
close(Writer);

my @got = <Reader>;
close(Reader);
waitpid($pid, 0);

print "Output got is: ";
print @got;

如您所见,我在开始阅读之前关闭了编写器,因为我的 C 程序旨在获取所有输入然后进行输出。现在整个进程间通信将起作用:

./program.pl
Output got is: Size is 60

作为旁注,您不需要关闭 $pid,因为它只是一个代表子进程 ID 的数字。在其他情况下,您可能想探索 non blocking reads ,但会使逻辑更加复杂。

原始答案:没有解决海报问题,因为他想使用 IPC。

您可以将文件名作为 sample.xml 添加到 $cmd 吗?您可以只使用反引号运算符来捕获输出,chomp 删除换行符并且输出将在 $out 中。

#!/usr/bin/perl
$cmd = './a.out < sample.xml 2> /dev/null';
$out = `$cmd`;
chomp $out;
print "out='$out'\n";

我猜你的例子是找到一种在 C 和 Perl 之间进行通信的方法,因为当然,如果它只是为了文件大小,那么在 Perl 中要容易得多:

#!/usr/bin/perl
print -s 'sample.xml';
print "\n";

关于c - Perl 将数据传递给 stdin 并从 stdout 和 stderr 读取,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7635298/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com