簡単な質問で恐縮ですが、現在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も見ています
-
餃子を食べるとき、何をつけますか?
みんな大好き餃子。 ふと素朴な疑問ですが、餃子には何をつけて食べますか? 王道は醤油とお酢でしょうか。
-
大人になっても苦手な食べ物、ありますか?
大人になっても、我慢してもどうしても食べれないほど苦手なものってありますよね。 あなたにとっての今でもどうしても苦手なものはなんですか?
-
とっておきの手土産を教えて
お呼ばれの時や、ちょっとした頂き物のお礼にと何かと必要なのに 自分のセレクトだとついマンネリ化してしまう手土産。 ¥5,000以内で手土産を用意するとしたらあなたは何を用意しますか??
-
許せない心理テスト
私は「あなたの目の前にケーキがあります。ろうそくは何本刺さっていますか」と言われ「12本」と答えたら「ろうそくの数はあなたが好きな人の数です」と言われ浮気者扱いされたことをいまだに根に持っています。
-
ハマっている「お菓子」を教えて!
この世には、おいしいお菓子がありすぎて……。 次何を食べたらいいか迷っています。 みなさんが今、ハマっている「お菓子」を教えてください!
-
コマンドプロンプトからのEOFの入力方法について
C言語・C++・C#
-
CTRL+Dでループを抜けるには
C言語・C++・C#
-
<unistd.h>をVisualStudioでつかえるようにする
C言語・C++・C#
-
-
4
C言語で今まで表示していた画面の消すには?
C言語・C++・C#
-
5
C言語 exitの使い方
C言語・C++・C#
-
6
終了条件Ctrl+zについて,結果表示ついて(C言語)
C言語・C++・C#
関連するカテゴリからQ&Aを探す
おすすめ情報
- ・漫画をレンタルでお得に読める!
- ・プリン+醤油=ウニみたいな組み合わせメニューを教えて!
- ・タイムマシーンがあったら、過去と未来どちらに行く?
- ・遅刻の「言い訳」選手権
- ・【大喜利】【投稿~11/12】 急に朝起こしてきた母親に言われた一言とは?
- ・好きな和訳タイトルを教えてください
- ・うちのカレーにはこれが入ってる!って食材ありますか?
- ・好きな「お肉」は?
- ・あなたは何にトキメキますか?
- ・おすすめのモーニング・朝食メニューを教えて!
- ・「覚え間違い」を教えてください!
- ・とっておきの手土産を教えて
- ・「平成」を感じるもの
- ・秘密基地、どこに作った?
- ・【お題】NEW演歌
- ・カンパ〜イ!←最初の1杯目、なに頼む?
- ・一回も披露したことのない豆知識
- ・これ何て呼びますか
- ・初めて自分の家と他人の家が違う、と意識した時
- ・「これはヤバかったな」という遅刻エピソード
- ・これ何て呼びますか Part2
- ・許せない心理テスト
- ・この人頭いいなと思ったエピソード
- ・牛、豚、鶏、どれか一つ食べられなくなるとしたら?
- ・ハマっている「お菓子」を教えて!
- ・【大喜利】【投稿~11/1】 存在しそうで存在しないモノマネ芸人の名前を教えてください
- ・好きなおでんの具材ドラフト会議しましょう
- ・餃子を食べるとき、何をつけますか?
- ・あなたの「必」の書き順を教えてください
- ・ギリギリ行けるお一人様のライン
- ・10代と話して驚いたこと
- ・つい集めてしまうものはなんですか?
- ・自分のセンスや笑いの好みに影響を受けた作品を教えて
- ・【お題】引っかけ問題(締め切り10月27日(日)23時)
- ・大人になっても苦手な食べ物、ありますか?
- ・14歳の自分に衝撃の事実を告げてください
- ・【大喜利】【投稿~10/21(月)】買ったばかりの自転車を分解してひと言
- ・ホテルを選ぶとき、これだけは譲れない条件TOP3は?
- ・家・車以外で、人生で一番奮発した買い物
- ・人生最悪の忘れ物
- ・【コナン30周年】嘘でしょ!?と思った○○周年を教えて【ハルヒ20周年】
- ・あなたの習慣について教えてください!!
- ・都道府県穴埋めゲーム
このQ&Aを見た人がよく見るQ&A
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
プログラミング初心者です。 Py...
-
正負を反転させて出力するプロ...
-
*をユーザーが入力した数字の数...
-
VisualStudio2019のコードアナ...
-
getchar()について 教えてくだ...
-
Linuxで入力待ちなしkeyread関...
-
C-Builderで数値(数字)以外を入...
-
ワードで文字を入力する時の変...
-
fgetsを用いたループ処理後の入...
-
通信対戦ゲームでの通信遅延に...
-
数値の連続入力終了条件について
-
applescript システム環境設定...
-
VBでの日付入力値のフォーマッ...
-
scanf関数について
-
"scanf"でエンターで改行させな...
-
C言語でgetchar();が上手く使え...
-
C++。社員データ入出力課題。
-
2進数の1の数を数える問題
-
日付型のデータに変換するには!?
-
漢字のソートについて
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
プログラミング初心者です。 Py...
-
正負を反転させて出力するプロ...
-
*をユーザーが入力した数字の数...
-
数字以外が入力されたらエラー...
-
Eclipseコンソール表示を、リセ...
-
scanfが2回使えない・・・?;
-
java初心者です。入力されたの...
-
Excel VBAで、Application.Inpu...
-
ワードで文字を入力する時の変...
-
WindowsでEOF
-
batプログラム上で文字列を入力...
-
getchar()について 教えてくだ...
-
cout関数を使っているのですが...
-
コマンドプロンプトからのEOFの...
-
"scanf"でエンターで改行させな...
-
VisualStudio2019のコードアナ...
-
至急教えてください!プログラ...
-
C言語scanf_sで何故か2回入力に...
-
scanf が無視されます
-
EDITコントロールで入力できる...
おすすめ情報