質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

Q&A

解決済

1回答

849閲覧

C++でPythonプログラムを呼び出した際の標準出力、標準エラー出力を変数に格納したい

kurage

総合スコア18

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

0グッド

0クリップ

投稿2022/06/11 07:47

1. 実現したいこと

C++のプログラムから呼び出したPython プログラムの標準出力、標準エラー出力を変数に格納したいと考えています。
変数に格納する方法について、ご教示ください。

2. 該当のソースコード

cpp

1#include <Python.h> 2#include <iostream> 3 4using namespace std; 5 6int main() 7{ 8 char mybuf[8192]; 9 10 /* printf() displays the string inside quotation */ 11 Py_Initialize(); 12 setvbuf(stdout, mybuf, _IOFBF, sizeof(mybuf)); 13 FILE *exp_file = fopen("./test.py", "r"); 14 PyRun_SimpleFile(exp_file, "./test.py"); 15 PyObject * module = PyImport_AddModule("__main__"); 16 PyObject *result = PyObject_CallObject(module, NULL); 17 cout << mybuf << endl; 18 fflush(stdout); 19 printf("Hello, World!"); 20 return 0; 21}

python

1def test(): 2 try: 3 // 処理 4 except Exception as e: 5 print('エラーが発生しました', file=sys.stderr) 6 sys.exit(1) 7 8if __name__ == '__main__': 9 test() 10 print("success main", file=sys.stdout)

3. 自分で調べたことや試したこと

標準出力の内容をmybufに格納できると思ったのですが、Pythonプログラムから出力している標準出力の内容は、変数mybufに格納されず、コンソールに出力がされました。

4. 使っているツールのバージョンなど補足情報

環境:CentOS7
※ Docker imageを使用
Python:3.6.8
GCC:4.8.5

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

python の print() は printf(3) でなく write(2) で出力しているので、setvbuf は使えません。
以下のように pipeを作り、標準出力を一時的にpipeに切り替える方法なら、python の print() の出力を受け取ることはできました。

#include <Python.h> #include <iostream> #include <unistd.h> using namespace std; int main() { int out_pipe[2]; int saved = dup(STDOUT_FILENO); pipe(out_pipe); dup2(out_pipe[1], STDOUT_FILENO); char mybuf[8192] = {}; /* printf() displays the string inside quotation */ Py_Initialize(); //setvbuf(stdout, mybuf, _IOFBF, sizeof(mybuf)); FILE *exp_file = fopen("./test.py", "r"); PyRun_SimpleFile(exp_file, "./test.py"); PyObject * module = PyImport_AddModule("__main__"); PyObject *result = PyObject_CallObject(module, NULL); read(out_pipe[0], mybuf, sizeof(mybuf)); dup2(saved, STDOUT_FILENO); printf("output is \"%s\"\n", mybuf); printf("Hello, World!"); return 0; }
# ./a.out output is "success main " Hello, World!

このコードはpipe のバッファサイズ以上の出力を想定していないので注意。念のため

投稿2022/06/11 23:41

sigsegv

総合スコア895

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

kurage

2022/06/12 07:07

ご教示いただきまして、ありがとうございます。 記載いただいた内容で期待の動作をさせることができました。 また、setvbufで値を取得することができない理由についても理解することができ、大変勉強になりました。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問