簡単な質問で恐縮ですが、現在WindowsでVC++2010Expressでコードを書いています。
そこで、以下の様な入力文字数を返すコードを書いているのですが、コマンドプロンプト上でCtrl+Z(Ctrl+Dも試した)を押してもストップしてくれません。(コマンドプロンプト上に「^Z(^D)」と表示されるだけです。
どなたか教えて頂けませんでしょうか?
#include <stdio.h>
int main()
{
int strText;
int counter=0;
while( (strText=getchar() ) != EOF){
++counter;
}
putchar(counter);
return 0;
}
No.7ベストアンサー
- 回答日時:
>私もそれが気になって「BackSpace」や「Delete」も試してみましたが、カウントされずに表示通りの文字数が返って来ました。
>この辺の動作が不明なのですが、「getchar()がEOF以外の値を返却した回数のカウントとなっているハズ」ではないのでしょうか?
標準入力は、通常では「バッファリング」されているので、改行がストリームを通過しないとプログラムに制御が戻りません。
「A」「B」「C」「バックスペース」「バックスペース」「^Z」「^Z」「1」「2」「3」と押しても、プログラムに返ってきません。
最後に「Enter」を押した瞬間、バッファ内の文字が整理されてから、プログラムに返って来ます。
上記の場合では、バッファ内が「A」「^Z」「^Z」「1」「2」「3」「\n」に整理されてから、プログラムに制御が戻ります。
「B」「C」「バックスペース2個」は、バッファが整理された時に削除されてしまいます。
すると、最初のgetcharには「A」が返されます。
その「A」は「EOFではない」ので、ループを繰り返します。
次にあるのは「^Z」ですが、これは「テキストモードでのファイルの終わり」なので、getcharは「ストリームが尽きた」と処理して「EOF」を返します。
この「EOF」は、通常は「文字コードに無い値」に定義されています(通常は「-1」になっている筈)
文字コードは「0~255」の筈ですから、どんな文字を入れても「-1」とは一致しません。
getcharが「EOF」を返したので、貴方のプログラムはループを終了してしまいます。
結果、上記のように入力すると、結果は「1」になるでしょう。
これは、プログラムを以下のようにすると「実感」出来ます。
#include <stdio.h>
int main()
{
int strText;
int counter=0;
while( (strText=getchar() ) != EOF){
++counter;
printf("L:%d\n",counter);
}
printf("E:%d\n",counter);
return 0;
}
プログラムが起動したら
ABC「Enter」DEF「Enter」123ABC「BS」「BS」「^Z」「Enter」と入力してみましょう。
結果は
ABC
L:1
L:2
L:3
L:4
DEF
L:5
L:6
L:7
L:8
123A^Z
L:9
L:10
L:11
L:12
E:12
となる筈です。
「Enter」を押さない限りはプログラムに戻って来ないので、「A」「B」「C」を入力した段階では、ループの中にある
printf("L:%d\n",counter);
は実行されません。
そして「Enter」を押した瞬間、それまでの間にバッファに溜まった文字が「改行も1文字」としてgetcharから返って来ます。
なので「Enter」を入力した瞬間、「A」「B」「C」「\n」が順に連続で返って来て、ループ内の表示が4回連続で一気に行われます。
わざわざ懇切丁寧に解説して頂いてありがとうございます!すごくわかりやすかったです。
ところで、ご示し頂いたコードをそのまま試してみたのですが、
不思議なことに、
(前略)
123A^Z
L:9
L:10
L:11
L:12
L:13
となり、もう一度「^Z」「Enter」を押さねば「E:13」という出力が現れません。
これは一体どういうことなんでしょうか??
No.14
- 回答日時:
>こういうことってなかなか入門書なんかには書いてないので非常にありがたいです!
入門書レベルではそこまで深い(環境依存度の高い)ネタはやらないんでしょう。
# たとえば質問でも書かれているCtrl+DでEOFはUNIX系の場合(というかシェル?)依存ですし。
>本にはこうなりますってものと同じもの書いてもならないと、本が間違ってるのか何のか分からなくなりますからね。
著者が前提としている(あるいは動作確認した)環境が明示されていないモノは…避けた方がいいかもしれませんね。
たいていはOSやコンパイラ、バージョンなんかが明記されているかと。
最初の方に書かれているか、後ろの方の索引近くに書かれているかという違いはありますが。
# 入門書なり購入するときは明記されているか確認・納得の上で購入した方がよいでしょう。
No.13
- 回答日時:
>ところで、ご示し頂いたコードをそのまま試してみたのですが、
>不思議なことに、
>(略)
>となり、もう一度「^Z」「Enter」を押さねば「E:13」という出力が現れません。
>これは一体どういうことなんでしょうか??
「^Z」の扱いについては、一部「環境依存」の部分があって、他の回答にある通り
「行頭での^ZのみEOFが返り、行頭ではない^Zは\x1aが返る」
と言う処理系と
「行頭でも、行頭以外でも^Zが現れた時点でEOFが返る」
と言う処理系があります。
なので、Visual Studioのデバッグ画面で試さず、リリース版exeを作ってからコマンドプロンプトで実行して試すと、結果が変わる可能性があります。
因みに、当方が試した環境は、後者の「^Zが現れた時点でEOFが返る」のタイプでした。
なるほど~。色々と環境の影響があるんですね。。
こういうことってなかなか入門書なんかには書いてないので非常にありがたいです!
本にはこうなりますってものと同じもの書いてもならないと、本が間違ってるのか何のか分からなくなりますからね。
No.12
- 回答日時:
#7へのお礼の中の疑問
> もう一度「^Z」「Enter」を押さねば「E:13」という出力が現れません。
OSの仕様か、Visual Studioの仕様かわかりませんが、Ctrl+Zが入力行の先頭にないとEOF(値は-1)は返ってきません。行の途中に入れると値0x1aが返ってきます。また、Ctrl+Zに続けて文字を入力させようとしても、その行のCtrl+Zより後の文字は改行文字を含めて破棄されてしまいます。
No.11
- 回答日時:
>その辺を制御したいならOS依存になるはずですがWindowsだとどうするんだろ。
そこまで行くとWindowsAPIでしょうか……。
http://msdn.microsoft.com/ja-jp/library/cc429744 …
なんての見つけましたけど……。
GetStdHandle()でハンドルを取得、GetConsoleMode()で現在のモードを取得して、
ENABLE_LINE_INPUTのフラグを落としてSetConsoleMode()で設定…でしょうか……。
APIで変更した内容がCランタイムライブラリの方に影響を与えるかどうか?に関しては疑問ですが。
# ランタイムの方の行バッファも無効にする必要がありますかねぇ…。
って、置いてけぼりとか言いつつ続けてしまう私。
# さすがに試してまではいない。
# コンソール上で動作するゲームでも作らないと行バッファとか問題にならない…でしょうし。
No.10
- 回答日時:
バックスペースやデリートで編集できるのは、(Cライブラリの)標準入力の行バッファリングではなく、OS側の行バッファリングの話かと。
なので、その辺を制御したいならOS依存になるはずですがWindowsだとどうするんだろ。
No.9
- 回答日時:
>それとも, VisualStudio の統合環境から?
でした。
というか面倒だったのでreturnにブレークポイント置いた。
まぁ本筋から逸れていってます……かね?
コンソール(CUI)使うことはほとんど無いしなぁ。
# 最近はC#使うことが多いし。
なんか質問者さん置いてけぼりにしている感が……。
No.8
- 回答日時:
OS レベルでバッファリングされると, そうなっちゃいますね>#6. もともとそこは C言語レベルではど~しよ~もないわけで, 実行環境に依存した方法を使わざるを得ない. 例えば, Windows 環境なら conio.h とか?
あと, どうやってプログラムを実行していますか? コマンドプロンプトから実行してる? それとも, VisualStudio の統合環境から? はたまた, それら以外の何か?
No.6
- 回答日時:
どうやらそのようで…>#5
#include <stdio.h>
int main()
{
int strText;
int counter=0;
while( (strText=getchar() ) != EOF){
++counter;
}
printf("\nCount=%d\n", counter);
setvbuf( stdin, NULL, _IONBF, BUFSIZ );
counter=0;
while( (strText=getchar() ) != EOF){
++counter;
}
printf("\nCount=%d\n", counter);
return 0;
}
とか、書いて試してみましたが動作は変わらず。
http://support.microsoft.com/kb/45563/ja
コンソール相手だと効かない…ってことですかねぇ。
# リダイレクトは試していない。
そんな訳で、#4はちょっと的外れな回答となってしまいました。
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
似たような質問が見つかりました
- C言語・C++・C# c言語 プログラムのエラー 1 2023/02/11 20:31
- C言語・C++・C# 宣言する関数の形が決まっている状態で、 str1とstr2の文字列をこの順に引っ付けてstrに保存し 2 2022/05/30 18:21
- C言語・C++・C# c言語配列の結合についてです。 なぜうまくいかないのでしょうか。 #include <stdio.h 4 2022/05/30 22:42
- C言語・C++・C# c言語の問題です 3 2023/01/10 16:15
- C言語・C++・C# プログラミングの授業の課題です 1 2023/01/17 22:15
- C言語・C++・C# c言語でユーザ関数を利用して入力された文字列を反転させるプログラムを作りたいです。 3 2023/01/29 19:47
- C言語・C++・C# c言語の問題の説明、各所ごとに 5 2023/07/26 11:03
- C言語・C++・C# バイナリファイルをコピーするのにかかる時間を測りたいのですが実行するとFatel error:gli 2 2022/11/03 01:10
- C言語・C++・C# char string[100]; int c, i=0; while( (c=getchar()) 2 2022/05/30 21:41
- C言語・C++・C# C pointer? or... 2 2022/03/29 00:47
このQ&Aを見た人はこんなQ&Aも見ています
-
これまでで一番「情けなかったとき」はいつですか?
これまでの人生で一番「情けない」と感じていたときはいつですか? そこからどう変化していきましたか?
-
一番好きな「クリスマスソング」は?
街に出ればクリスマスソングを聞かない日はないくらい、 いろんな曲がかかっていますよね。 あなたが一番好きな「クリスマスソング」を教えてください!
-
2024年に成し遂げたこと
今年も残すところわずか。 皆さんが今年達成したことを教えていただきたいです。 どんな小さなものでも構いません。
-
何歳が一番楽しかった?
自分の人生を振り返ったとき、何歳のころが一番楽しかったですか? 子供の頃でしょうか、それとも大人になってからでしょうか。
-
冬の健康法を教えて!
温度変化が大きくなり、風邪をひきやすいこれからの季節。 どんなことに気をつけていますか?
-
コマンドプロンプトからのEOFの入力方法について
C言語・C++・C#
-
CTRL+Dでループを抜けるには
C言語・C++・C#
-
<unistd.h>をVisualStudioでつかえるようにする
C言語・C++・C#
-
-
4
csvファイルの最後にeof(0x1a)を付けたいのですが
Windows Vista・XP
-
5
C言語で今まで表示していた画面の消すには?
C言語・C++・C#
-
6
C言語 exitの使い方
C言語・C++・C#
-
7
終了条件Ctrl+zについて,結果表示ついて(C言語)
C言語・C++・C#
関連するカテゴリからQ&Aを探す
おすすめ情報
- ・「みんな教えて! 選手権!!」開催のお知らせ
- ・漫画をレンタルでお得に読める!
- ・【選手権お題その2】この漫画の2コマ目を考えてください
- ・2024年に成し遂げたこと
- ・3分あったら何をしますか?
- ・何歳が一番楽しかった?
- ・治せない「クセ」を教えてください
- ・【大喜利】看板の文字を埋めてください
- ・【大喜利】【投稿~12/17】 ありそうだけど絶対に無いことわざ
- ・【選手権お題その1】これってもしかして自分だけかもしれないな…と思うあるあるを教えてください
- ・【穴埋めお題】恐竜の新説
- ・我がまちの「給食」自慢を聞かせてっ!
- ・冬の健康法を教えて!
- ・一番好きな「クリスマスソング」は?
- ・集合写真、どこに映る?
- ・自分の通っていた小学校のあるある
- ・フォントについて教えてください!
- ・これが怖いの自分だけ?というものありますか?
- ・スマホに会話を聞かれているな!?と思ったことありますか?
- ・それもChatGPT!?と驚いた使用方法を教えてください
- ・見学に行くとしたら【天国】と【地獄】どっち?
- ・これまでで一番「情けなかったとき」はいつですか?
- ・この人頭いいなと思ったエピソード
- ・あなたの「必」の書き順を教えてください
- ・10代と話して驚いたこと
- ・14歳の自分に衝撃の事実を告げてください
- ・人生最悪の忘れ物
- ・あなたの習慣について教えてください!!
- ・都道府県穴埋めゲーム
このQ&Aを見た人がよく見るQ&A
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
入力値が1以下、かつ数字以外の...
-
コンソールアプリの入力時に改...
-
数字以外が入力されたらエラー...
-
Userformの入力順序をタブオー...
-
C-Builderで数値(数字)以外を入...
-
物理の斜方投射のシミュレーシ...
-
batプログラム上で文字列を入力...
-
コマンドライン上で確定した文...
-
C言語 While文(ループ)内の...
-
Excel VBAで、Application.Inpu...
-
C言語・YesNo入力のループで解...
-
Eclipseでコマンドラインを入力...
-
至急教えてください。プログラ...
-
C++:cinが上手く使えない
-
正負を反転させて出力するプロ...
-
コマンドプロンプトからのEOFの...
-
*をユーザーが入力した数字の数...
-
EDITコントロールで入力できる...
-
2進数の1の数を数える問題
-
キーボード入力によるループの終了
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
プログラミング初心者です。 Py...
-
正負を反転させて出力するプロ...
-
*をユーザーが入力した数字の数...
-
数字以外が入力されたらエラー...
-
コマンドプロンプトからのEOFの...
-
Eclipseコンソール表示を、リセ...
-
Excel VBAで、Application.Inpu...
-
"scanf"でエンターで改行させな...
-
cout関数を使っているのですが...
-
EDITコントロールで入力できる...
-
scanfが2回使えない・・・?;
-
C言語scanf_sで何故か2回入力に...
-
getchar()について 教えてくだ...
-
if文の条件にscanf関数を使うと…?
-
漢字のソートについて
-
プログラミングの問題です 「金...
-
VB.NETで16進数+16進数や16進...
-
VisualStudio2019のコードアナ...
-
入力候補を表示させるには・・・?
-
実行結果の順番がおかしいんで...
おすすめ情報