
下記のc++のcinの動作について質問があります。分かる方、教えてください。
コンパイルして実行後、標準入力から、3.14yと入力すると、
val = 3.39
ch = y
の期待通りの結果になるのですが、
3.14xと入力すると、
val = 0.25
ch = 3
の結果になってしまいます。
調べて見ると、最後に入れる数値以外の文字(この場合の、xやy)が、文字によってdoubleと認識されて取り込まれる場合があるようで、その場合取り込まれた文字がdoubleに変換されて0になって、入力ストリームが最初の位置に戻されるようなのですが、これは正しい動作なのでしょうか?それとも書き方に何かまずい箇所とかあるのでしょうか?
#include <iostream>
using namespace std;
int main() {
char ch;
cin >> ch;
switch (ch) {
case '.':
case '0': case '1': case '2': case '3': case '4':
case '5': case '6': case '7': case '8': case '9':
{
//cout << "here" << endl;
cin.putback(ch);
double val;
cin >> val;
val += 0.25;
cout << "val = " << val << endl;
}
default:
cin >> ch;
cout << "ch = " << ch << endl;
}
return 0;
}
A 回答 (5件)
- 最新から表示
- 回答順に表示
No.5
- 回答日時:
手元の WSL では g++ 6.3.0 でも想定した通りの動作になっているから, おそらく「とてつもなく古い g++」でない限りは「プログラムとしての問題」はないと思うよ.
どこかのタイミングで failbit が立ってたりしないかなぁ. 「入力ストリームが最初の位置に戻される」ということではないはずだけど, あなたのいう症状が手元では全く出てこないので完全に推測の域を出ない.
No.4
- 回答日時:
確認だけど, このプログラムだと「3.14x」とか「3.14y」とかを 1回入力したらプログラムそのものも終了するよね. それを前提に
・まずこのプログラムを実行し, そこで「3.14y」と入力すると
val = 3.39
ch = y
と出力されてプログラムが終了する
・その次に改めてこのプログラムを実行し, そこで「3.14x」と入力すると
val = 0.25
ch = 3
と出力されてプログラムが終了する
という動作になっていて困っている, ということでいい?
No.3
- 回答日時:
回答になっていませんが、逆の順番にやってみたらどうなります?
つまり、
標準入力から、3.14xと入力すると、
val = 3.39
ch = x
の期待通りの結果になるのですが、
3.14yと入力すると、
val = 0.25
ch = 3
の結果になってしまいます
ということはないでしょうか?

No.1
- 回答日時:
人工知能はこう答えたけど、私はプログラミングのことはわかりません。
-----------
このコードでは、最初に入力された文字を ch として読み込み、その文字が '.' もしくは数字の場合に限り、次に double 型の変数 val を読み込んでいます。この val に対して 0.25 を加算し、その結果を出力しています。それ以外の場合には、もう一度文字を読み込んで ch に格納し、その結果を出力しています。
「3.14y」という入力に対しては、最初に '3' が ch に格納され、その後 cin.putback(ch) を使って '3' を入力ストリームに戻しています。次に cin >> val が実行され、3.14 が val に格納されます。最後に ch には 'y' が読み込まれ、それが出力されます。
一方、「3.14x」という入力に対しては、最初に '3' が ch に格納されますが、次に '.' が読み込まれるので、cin.putback(ch) を使って '3' を入力ストリームに戻すことはできません。そのため、cin >> val が実行されると、val には 0.14 が格納されます。その後、'x' が ch に読み込まれ、それが出力されます。
つまり、このコードでは最初に入力された文字が数字や '.' である場合に限り double 型の変数に変換し、それ以外の場合は単に文字として読み込んでいるため、入力によっては意図しない結果になる可能性があります。
もし double 型の数値を正確に読み込む必要がある場合には、より厳密な入力処理を行う必要があります。例えば、std::getline を使って一行を読み込み、それを std::istringstream に渡して double 型に変換する方法があります。
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
関連するカテゴリからQ&Aを探す
おすすめ情報
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
至急お願いします。プログラミ...
-
getcharをしながらwhile文で出...
-
C言語でgetchar();が上手く使え...
-
数字以外が入力されたらエラー...
-
正負を反転させて出力するプロ...
-
小数か整数かを判定する方法
-
[C#]順序通りに実行されない
-
C++ scanfで止まらない
-
*をユーザーが入力した数字の数...
-
c言語 ボタン入力のプログラム...
-
プログラミング初心者です。 Py...
-
BM法(ボイヤームーア法)に...
-
Excel VBAで、Application.Inpu...
-
scanf が無視されます
-
8086のアセンブリで文字列の入...
-
入力チェックのプログラム
-
バッチファイルで、キーボード...
-
16進数を入力するプログラムを...
-
MinGWのC言語でCTRL+Zで処理が...
-
iPhoneキーボードの右下の「開...
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
正負を反転させて出力するプロ...
-
プログラミング初心者です。 Py...
-
数字以外が入力されたらエラー...
-
Excel VBAで、Application.Inpu...
-
EDITコントロールで入力できる...
-
*をユーザーが入力した数字の数...
-
Linuxプログラミングで、キーボ...
-
Eclipseコンソール表示を、リセ...
-
入力候補を表示させるには・・・?
-
batプログラム上で文字列を入力...
-
UWSCで変数をキー入力
-
VisualStudio2019のコードアナ...
-
小数か整数かを判定する方法
-
scanfが2回使えない・・・?;
-
Linuxで入力待ちなしkeyread関...
-
java初心者です。入力されたの...
-
Delphi初心者 ボタン操作につ...
-
コマンドプロンプトからのEOFの...
-
Eclipseでコマンドラインを入力...
-
Userformの入力順序をタブオー...
おすすめ情報
g++は、12.0.0
clang++は、12.0.0
で行っています。
結果は、上に書いた通りです。
そうです。
順番は関係ないです。
最後の文字が何であるかだけでの再現性です。(うまくいくものはうまくいくし、うまくいかないものはうまくいかない。)
ちなみに、3.14の後にリターンを押して、それから文字を入れる場合はOKです。
また、doubleをintに変えてやった場合は、文字を続けて入れた場合、リターンして入れた場合、両方ともOKになります。
こちらでも確認してみたのですが、
WindowsとLinuxの環境でのg++では共に、OKみたいです。
ただ、macはiMacとMacBook Pro共に上に書いたような状態になります。
ちなみにiMacのg++のヴァージョンは、13.0.0です。
macの環境でもうまくいってる人っています?
何か私のmacの環境でおかしなところがあるのかもしれないので^^;
後、macでは、
g++は実際には、clang++が動いているみたいです。
g++とclang++の差かもしれません。