リダイレクトとパイプについて教えてください。
test.exeというコンソールプログラムがあり、
実行すると、1秒置きに10回コンソール画面に"!!!!!!"を表示して終了するプログラムとします。
これをリダイレクトで
test.exe > result.txt
とした場合、1秒置きに"!!!!!!"がresult.txtに追加されることを期待しているのですが、
10秒後にtest.exe終了時に一気に、"!!!!!!"が10個吐き出されます。
1秒おきに(リアルタイムで)、"!!!!!!"を吐き出すようにはできないでしょうか。
標準入力を受け取ってファイルに吐き出すプログラムresult.exeをつくり、
これをパイプで
test.exe | result.exe
としても、リアルタイム吐き出しができず、同じ結果でした。
No.6ベストアンサー
- 回答日時:
コンソール直出力の場合はバッファが無いので、即座に
表示されます。リダイレクトすると、出力先がファイル
なので、バッファリングされます。
パイプの場合はAPIでやると分かるのですが、ファイルと
同じ扱いになります。(1度のOpen、正しくはCreateで
入力と出力の2個のハンドルができる所が異なる)つまり、
バッファリングされます。
デバイスの種類からみると、コンソールはキャラクタ型の
デバイスで、最小単位が1バイトであるため、1文字でも
出力すれば装置にも反映されます。
これに対し、ファイル(ディスク)はブロック型デバイス
になります。最小入出力単位はセクタサイズになります。
つまり、512バイト(だったと思う。間違ってたらゴメン)
単位でしか入出力できません。これを1バイト毎に反映
しようとすると、セクタの読み込み→変更→書き込み という
動作を1バイト毎にしなければなりません。
これではディスクIOの回数が膨大なものになり、効率が
極端に悪くなります。それでバッファリングする訳です。
ついでに言うと、1セクタのIOと、1シリンダのIOの時間は
あまり変わりません。
詳しい説明、ありがとうございます。
説明に基づき、あえてバッファーが一杯になるように、
沢山の!!!!!!!を表示する実験をしてみました。
printf("!!!!!!!!!!!!!!!!!!!!!!!!!・・・・・・・・・・・!!!!!!!!!!!!!!!!!!!!!\n");
すると、test.exeの実行途中(10秒にならない間でも)
リダイレクトされることが確認できました。
やはり、バッファに入ったままなのですね。
今回実際にやろうとしたことは、icl.exe(インテルC++コンソールコンパイラー)のGUI化です。
Windowsプログラムから、icl.exeをCreateProcessで動かして、コンパイル過程やエラー表示などを
リダイレクトして、Windowsプログラム上に表示させようとしていました。
コンパイルで時間がかかる場合も、最後まで待たずに表示したいと思っていました。
ちょっと無理そうですね。
でも、無理な理由が分かってすっきりしました。
ありがとうございました。
No.8
- 回答日時:
>今回実際にやろうとしたことは、icl.exe(インテルC++コンソールコンパイラー)のGUI化です。
>Windowsプログラムから、icl.exeをCreateProcessで動かして、コンパイル過程やエラー表示などを
>リダイレクトして、Windowsプログラム上に表示させようとしていました。
>コンパイルで時間がかかる場合も、最後まで待たずに表示したいと思っていました。
参考程度に、
こういうのはいかがでしょう
http://www.atmarkit.co.jp/fdotnet/dotnettips/657 …
やった事が無いので、出来るかどうかは知りません
参考URL:http://www.atmarkit.co.jp/fdotnet/dotnettips/657 …
ありがごうございます。
リダイレクトが明示的に行える関数が存在するのですね。
icl.exeの出力がバッファにたまるかや、
icl.exeが実行途中でも非同期に出力を確認できるかは、
やってみないとわかりませんね。
参考にさせて頂きます。
No.7
- 回答日時:
出力が何メガバイトでもファイルが閉じられるまでは内容は更新されないですよね
ディスクにデータが書き込まれてもそれが直ちにファイルシステム上でファイル名と関連づけられるわけではないと理解しています
リダイレクト先のファイルが閉じるのは元の実行ファイルが終了するときなので結局リダイレクト元のプログラム終了までファイルの内容は更新されないと言うことではないでしょうか
この回答への補足
書き込みありがとうございます。
以下にオーバーフローさせる実験を書かせて頂きました。
今回は、面白い実験をさせて頂きました。
皆様ありがとうございました。
私の認識ミスや、もっとこうすれば、などありましたら、
今後もよろしくお願いします。
No.5
- 回答日時:
No.3です。
誤りがありましたので訂正します。
コンソール出力でもバッファリングされますが、その場合は、'\n'(改行コード)で出力される仕様になっています。
この回答への補足
test.exeはあくまで例で、このプログラムは”変更できない”が拘束条件として質問させて下さい。
例えば、gcc.exeなど、外部プログラムを使う場合などと思って下さい。
よって、C言語のファイル出力ではなく、
「リダイレクト」について、教えて頂ければと思っています。
今回、リダイレクトの実験のために、以下のソースをコンパイルし、test.exeとしました。
int main()
{
for(int i=0;i<10;i++){
printf("!!!!!!\n");
sleep(1);
}
return 0;
}
改行コードが入りですので、バッファリングすることなく、コンソール画面に"!!!!!!"が表示されます。
しかし、これをリダイレクトしても、ファイルへは10秒後にしか反映されませんでした。
「それは仕様で、方法はありません」が答えでも、もちろんOKです。
設定を変えたり、他のコマンドやオプションで変更できる方法があれば、
と思い、質問させて頂きました。
詳しく、何度も根気よく説明してくださりありがとうございました。
一人ひとりに的確なお礼がかけなくてすみません。
皆さんに感謝しています。
No.4
- 回答日時:
>test.exeは既存プログラムのため、変更できません。
あきらめてください。
おそらくprintf等の関数で出力していると思いますが、
バッファリングされている入出力の場合はファイルを
閉じるかflushするまで内部に保留された状態が続き
ます。どちらの手当もされていなければ、バッファが
溢れるか、標準出力が閉じられるまで、外部に出力
されることはありません。出てこないものを外部で
努力しても受け取ることはできません。
No.3
- 回答日時:
No.2です。
>test.exeの結果は、バッファにたまることなく
バッファリングされるかどうかは、出力する媒体によって違います。
画面上ではバッファリングされないだけです。
(もし疑うのならば、C言語の開発環境はあるようなので自分で試してみればいいでしょう)
ですから、test.exeを書き換えできないのならば、答えとしてはできないと言うことになります。
もしかしたら、システムをいじればできるのかも知れませんが、test.exeだけでなく他へも影響しますので、現実的とは言えないでしょう。
No.2
- 回答日時:
言語が書いてないので、正解ではないかも知れませんが、考え方を書きたいと思います。
C言語であれば
fflash(stdout);
を使って強制的にはき出します。
「バッファリング」と言われ、ディスクに出力する場合などは小まめにはき出されるのではなく、ある程度のサイズになるまでメモリー上に保存してから出力するようにします。
この方が効率良くディスクに書き込めるからです。
他の言語でも同じことをできるはずですので、調べるか言語を明示すれば具体的な解答が得られるでしょう。
参考URL:http://www9.plala.or.jp/sgwr-t/lib/fflush.html
この回答への補足
お返事ありがとうございます。
説明不足ですみません。
test.exeは既存プログラムのため、変更できません。
test.exeの結果は、バッファにたまることなく、コンソール画面に1秒置きに表示されています。
リダイレクトされるファイル自体は、空の状態ですぐに作成されますが、
中身が得られるのが、10秒後に一気に、という状態です。
No.1
- 回答日時:
text.exeを作った言語は何でしょう?
C言語であるなら、低水準入出力を使うことで、
リアルタイム出力が可能です。
for ( i = 0 ; i < 10 ; i++ ) {
//printf("%s\n", "!!!!!!"); //これはダメ
write(stdout->_file,"!!!!!!",6); //このようにする
Sleep(1000); //1秒待機
}
この回答への補足
お返事ありがとうございます。
また、説明不足ですみません。
test.exeは既存のプログラムで、私が変更できません。
なお、WindowsXP上のコンソールでの実行です。
reply.exeは、C言語で作りました。
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
似たような質問が見つかりました
- その他(プログラミング・Web制作) python OpenPyXLを使って出力結果をエクセルに書き込み 2 2022/06/04 19:46
- C言語・C++・C# 至急教えてください!プログラミングの問題です。 入力待ちをして、受け取った正の整数が表す行数だけ既存 4 2022/07/05 10:12
- Visual Basic(VBA) batからexeを実行し戻り値を受け取る EXEの実行内容の結果によって、戻り値を0か1かで返したい 1 2023/07/04 16:40
- Visual Basic(VBA) batからexeを実行し戻り値を受け取る バッチからEXEの結果を受け取りたいのですが、 下記のバッ 1 2023/07/04 15:13
- PHP PHP一覧表示した項目にリンクをはりたい 1 2023/07/12 17:08
- Windows 10 Windows11の実行モジュールはどこにある 1 2022/05/31 07:33
- C言語・C++・C# c言語の問題です 3 2023/01/10 16:15
- C言語・C++・C# exeファイルが作れない(windows10) 6 2022/08/13 08:47
- その他(プログラミング・Web制作) VScodeでpythonプログラムの関数を実行したい 2 2022/07/13 19:24
- その他(プログラミング・Web制作) pythonでクラスで複数のメソッドを利用する方法 2 2022/04/15 04:17
関連するカテゴリからQ&Aを探す
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
シェルコマンドの 2>&1 とはど...
-
C言語による10進数→16進数変換...
-
(VBA)書式が変更されてしまい...
-
C++ fprintf_sの使い方がわからん
-
AccessVBA複数レポート条件毎に...
-
C++/CLIにて、System.String^型...
-
ファイル形式またはファイル拡...
-
VBA でメモ帳へ保存する際の保...
-
アプリケーションのログファイ...
-
ファイル出力の場所を指定
-
stdin,stdoutについて
-
VB.NETでExcelファイルを出力す...
-
テキストファイルに改行コード...
-
C言語での印刷方法
-
printfだと出力されるのにfprin...
-
CBool関数について VB6とVB.net...
-
gccによって、BOM付きのUTF-8で...
-
C言語で16進数をテキスト出力し...
-
Base64でエンコードした後の文...
-
fortranのtxtファイル出力書式...
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
Paiza Cloudです。 どうやれば...
-
Paiza Cloudです。学籍番号と氏...
-
Acccess レポートをグループ別...
-
ファイル出力の場所を指定
-
コマンド(例えばls)の出力結果...
-
VC++でUTF-8のファイルを出力し...
-
テキストファイルに改行コード...
-
シェルコマンドの 2>&1 とはど...
-
VBA でメモ帳へ保存する際の保...
-
ファイル形式またはファイル拡...
-
fortranのtxtファイル出力書式...
-
CSV形式に変換
-
printfだと出力されるのにfprin...
-
C++ fprintf_sの使い方がわからん
-
c言語の質問です。 ランダムに4...
-
Wordマクロで指定したフォルダ...
-
C++/CLIにて、System.String^型...
-
pcap形式データをテキストへ抽出
-
二次元配列をクリップボードに...
-
raw形式からbmp形式への書き込...
おすすめ情報