从C/C++调用Objective C stdin/stdout管道时,它是如何工作的
How does Objective C stdin/stdout piping work when called from C/C++?
这是我整个下午都在处理的一个问题,我想我已经把它归结为核心问题,这似乎是在将数据传输到从C++应用程序调用的Objective C命令行应用程序时的意外行为。
当单独执行时,Objective C程序按预期工作。当C++管道(在这种情况下是"主",C++正在调用目标C可执行文件)调用类似于下面目标C代码的C/C++可执行文件时,一切也按预期工作。
此外,如果从Objective C中删除了输入代码,或者C++程序命令Objective C通过管道传输到一个文件(因此命令将为"./HelloWorld>dump.txt"而不是"./HelloWorld"),则一切都会按预期执行。
然而,当执行下面给出的代码时,在目标C尝试读取stdin之前的第一次尝试中,当试图读取目标C的stdout时,C++将挂起。
目标C
#import <Foundation/Foundation.h>
void c_print(NSString* prnt)
{
printf("%s", [prnt cStringUsingEncoding:NSUTF8StringEncoding]);
}
void c_print_ln(NSString* prnt)
{
printf("%sn", [prnt cStringUsingEncoding:NSUTF8StringEncoding]);
}
NSString* read_till(char c)
{
NSMutableString* ret = [[NSMutableString alloc] initWithString:@""];
char r = getchar();
while(r!=c && r!= ' ')
{
[ret appendFormat:@"%c",r];
r = getchar();
}
return ret;
}
int main(int argc, const char * argv[]) {
@autoreleasepool {
c_print_ln(@"Hello, World!");
NSString* exmp = read_till('n');
c_print_ln([[NSString alloc] initWithFormat:@"String I read: "%@"",exmp]);
}
return 0;
}
C++(.h文件)
#ifndef PIPE_H
#define PIPE_H
#include <sys/types.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <string>
#include <iostream>
#define PIPE_READ 0
#define PIPE_WRITE 1
class outsideExecutable
{
private:
char buf[1024];
bool is_good;
int infp, outfp;
public:
outsideExecutable(char* command);
~outsideExecutable();
bool isGood();
std::string readline();
void writeline(std::string source);
};
#endif
C++(.cpp文件)
#include "Pipe.h"
using namespace std;
int main()
{
cout<<"Testing Pipe"<<endl;
outsideExecutable* exe = new outsideExecutable((char*)"./HelloWorld");
exe->readline();
exe->writeline("reading example");
exe->readline();
delete exe;
}
static pid_t popen2(const char *command, int *infp, int *outfp)
{
int p_stdin[2], p_stdout[2];
pid_t pid;
if (pipe(p_stdin) != 0 || pipe(p_stdout) != 0)
return -1;
pid = fork();
if (pid < 0)
return pid;
else if (pid == 0)
{
close(p_stdin[PIPE_WRITE]);
dup2(p_stdin[PIPE_READ], PIPE_READ);
close(p_stdout[PIPE_READ]);
dup2(p_stdout[PIPE_WRITE], PIPE_WRITE);
execl("/bin/sh", "sh", "-c", command, NULL);
perror("execl");
exit(1);
}
if (infp == NULL)
close(p_stdin[PIPE_WRITE]);
else
*infp = p_stdin[PIPE_WRITE];
if (outfp == NULL)
close(p_stdout[PIPE_READ]);
else
*outfp = p_stdout[PIPE_READ];
return pid;
}
outsideExecutable::outsideExecutable(char* command)
{
is_good = false;
if (popen2(command, &infp, &outfp) <= 0)
return;
is_good = true;
}
outsideExecutable::~outsideExecutable()
{
}
bool outsideExecutable::isGood()
{
return is_good;
}
std::string outsideExecutable::readline()
{
if(!is_good)
return "";
string ret = "";
char hld;
read(outfp, &hld, 1);
while(hld!='n' && hld!=' ')
{
ret = ret + hld;
read(outfp, &hld, 1);
}
cout<<"We read:"<<ret<<endl;
return ret;
}
void outsideExecutable::writeline(std::string source)
{
if(!is_good)
return;
//Do nothing
cout<<"Sending command: "<<source<<endl;
source = source+"n";
write(infp, source.c_str(), source.length());
}
#endif
有人知道这可能出了什么问题吗?我在C/C++方面有相当多的经验,而且这方面的管道代码似乎运行得很好。这似乎真的是Objective C表现不好的一个例子,我从来没有见过这样失败的例子。
rich(https://stackoverflow.com/users/1566221/rici)刚刚在上面的评论中提供了这个问题的答案。以下是更新的目标C代码来修复它:
void c_print(NSString* prnt)
{
printf("%s", [prnt cStringUsingEncoding:NSUTF8StringEncoding]);
fflush(stdout);
}
void c_print_ln(NSString* prnt)
{
printf("%sn", [prnt cStringUsingEncoding:NSUTF8StringEncoding]);
fflush(stdout);
}
显然,为了使管道正常工作,需要在目标C中冲洗stdout。
相关文章:
- QSqlquery prepare()和bindvalue()不工作
- 导入库可以跨dll版本工作吗
- 以螺旋方式打印矩阵的程序.(工作不好)
- 对象指针在c++中是如何工作的
- 为什么在Windows上的VS 2019和Clang 9中"size_t"在没有标题的情况下工作
- VSOMEIP-2个设备之间的通信(TCP/UDP)不工作
- 为字符串中每 N 个字符插入空格的函数没有按照我认为的方式工作?
- C++为线程工作动态地分割例程
- 为什么我的 std::ref 无法按预期工作?
- 布尔比较运算符是如何在C++中工作的
- SampleConsensusPrerejective(ext.RANSAC)是如何真正工作的
- 不确定要在我的main中放入什么才能使我的代码正常工作
- 为什么std::condition_variable notify_all的工作速度比notify_one快(对于随机请
- <<操作员在下面的行中工作
- 有人能解释一下为什么下界是这样工作的吗C++的
- ExtractIconEx:可以工作,但偶尔会崩溃
- C++中的memset函数工作不正常
- 当我在第一个循环中使用"auto"时,它工作正常,但是使用"int"它会给出错误,为什么?
- 链表c++插入,所有情况都已检查,但没有任何工作
- 当 int 方法工作正常时,void 方法有何不同,或者为什么我不能调用 void 方法?