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

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

ただいまの
回答率

90.34%

  • C

    3965questions

    C言語は、1972年にAT&Tベル研究所の、デニス・リッチーが主体となって作成したプログラミング言語です。 B言語の後継言語として開発されたことからC言語と命名。そのため、表記法などはB言語やALGOLに近いとされています。 Cの拡張版であるC++言語とともに、現在世界中でもっとも普及されているプログラミング言語です。

  • for

    255questions

    for文は、様々なプログラミング言語で使われている制御構造です。for文に定義している条件から外れるまで、for文内の命令文を繰り返し実行します。

for文のインクリメント・条件判定のタイミングと型キャストの働き

解決済

回答 4

投稿

  • 評価
  • クリップ 5
  • VIEW 3,197

yusukee345

score 23

以下のような問題を考えます。
「forを用いて1.0から10.0までの値を0.1刻みで表示するコードを作成する。ループ制御には浮動小数点数を使い、条件式部分はキャストを用いて整数式とせよ。」
このようなコードを書いたら上手く実行できました。

#include<stdio.h>

void main(){
    double f;
    for (f = 1.0; (int)f < 10;f+=0.1){
        printf("%f ",f);
    }
    getchar();
}


ここで疑問があります。for文はfor(1;2;3){4}で1→243→243→243の順に実行されますよね。
このforループは
f=1.0→(中略)→9.90表示→9.90+0.1=10.0にインクリメント→(次のループへ)→(int)f<10の判定(10<10)→10.0表示→10.0+0.1=10.1にインクリメント→(次のループへ)→(int)f<10の判定→条件に合わずループ抜ける
という流れになっていますが、9.90+0.1=10.0にインクリメントした後は(int)fは10になるはずなので(int)f<10の条件をクリアできず10.0を表示出来ないはずです。でも実際には10.0は表示されているので条件はクリアできたことになります。
何が起こっているのかを教えて貰えないでしょうか。

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

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

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

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 4

checkベストアンサー

+9

こんにちは。

(赤文字は2進数表現です。その他は10進数表現です。)

0.1は2進数に変換すると、循環小数になってしまいます。
0 . 0001 1001 1001 1001 1001 1001 ...
C/C++のdoubleはIEEE754のbinary64で表現する処理系が多いですが、その場合 仮数部は53ビットあります。
1 . 1001 1001 1001 1001 1001 ...
という感じで表現されます。
53%4は3なので最後の1001のセットの10までで有効桁が終わり、その次が0ですから、四捨五入(零捨一入)され01 1001 ...が切り捨てられます。
微妙に0.1より小さい数字を100回足した結果は10より微妙に小さいため、(int)fで小数点下が切り捨てられ、条件を満たさないということになります。

(int)で切り捨てる前にround()で丸めると意図通りに動作する筈です。

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2017/03/10 00:29

    roundで丸めると、9.6の時点でループから抜けてしまうのでは?
    あと、9.99...は10より小さいのに、なぜ「条件文で」キャストしてあるのか、その説明があるとなお親切かと(9.99...9<10ですよね?)

    キャンセル

  • 2017/03/10 05:09

    > roundで丸めると、9.6の時点でループから抜けてしまうのでは?

    ああ、その通りですね。ご指摘ありがとうございます。

    (int)へのキャストはともともキャストされていたからですが、整数で比較した方が速いし、今回のようなトラブルが起きにくいので好ましいですね。

    yusukee345さん。
    10倍してroundして10で割るか、10倍してroundして100と比較するのがお手軽です。
    (int)(round(f*10.)/10.) < 10
    (int)round(f*10.) < 100

    四捨五入する桁を選べないって不便ですね。

    キャンセル

  • 2017/03/10 20:43

    回答ありがとうございます。浮動小数点数についてそのような根本的なことをあまり考えたことが無かったです。

    キャンセル

+6

理解としてはあっていますから大丈夫ですよ。
ソースも正しいと思います。

double型のfが10.0と思われるのにf<10の条件を抜けてしまうのは浮動小数点数型につきものの誤差が原因だと思います。
こちらのページが分かりやすいです。

double型で計算をしてしまうと、10.0に見えても実際には9.99999999999999999999999何がしだったりになってしまうことがあります。
そうするとint型に変換した時9になってしまいます。

会社かどこかの研修であれば、ちょっと話し合ってみると良いでしょう。
(僕ならこういう問題はそういう話すること前提じゃなきゃ出さないです)

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2017/03/10 20:48

    ありがとうございます。リンク先分かりやすいです。

    キャンセル

0

浮動小数点云々については他の方の回答で間に合ってると思いますので、ちょっと別の視点から。

1.0から10.0までの値を0.1刻みで表示するコードを作成する

通常の日本語で解釈すれば 1.0 ≦ x ≦ 10.0 と1と10を含むことになるので

for(f = 1.0; f <= 10.0; f+=0.1)

と組むのが自然です。

ただ、実際の組み方としては、他の方の指摘にもある通り浮動小数は数学的に正しく全ての数値を表現することが出来ないので、変数をforのループ変数として使うことは避けるべきです。
問題を解釈すると、「1.0~10.0までを0.1刻みで数値を表示する」ということなので、

int i;
double f = 1.0;
for(i = 0; i < 10; i++) {
    printf("%f ",f);
    f += 0.1;
}

とすれば、数値の初期値と増分がどういう値であっても確実に10個表示されることになります。(表示個数、初期値、増分を変数にして関数化してもいいと思います)

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2017/03/10 20:45

    ありがとうございます。キャストを使わないのならそのように書くのがベストでしょうね。

    キャンセル

-1

問題の方が浮動小数点数を指定しているので、出題者と話し合ってもらうのが良いと思ったんですよね。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2017/03/10 16:43

    私もそう思いますね。出題の意図が何かしらあると思われますからね。
    forのループ変数に浮動小数を使うことの問題点とどうすれば良いのかを考えさせるための出題かもしれません。

    キャンセル

  • 2017/03/10 20:47

    某C学習書からの問題ですが、出題意図はキャストをどう使うかというとこですね。ただ予想外の問題が隠れていて面白いなとも思います。

    キャンセル

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

  • ただいまの回答率 90.34%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

同じタグがついた質問を見る

  • C

    3965questions

    C言語は、1972年にAT&Tベル研究所の、デニス・リッチーが主体となって作成したプログラミング言語です。 B言語の後継言語として開発されたことからC言語と命名。そのため、表記法などはB言語やALGOLに近いとされています。 Cの拡張版であるC++言語とともに、現在世界中でもっとも普及されているプログラミング言語です。

  • for

    255questions

    for文は、様々なプログラミング言語で使われている制御構造です。for文に定義している条件から外れるまで、for文内の命令文を繰り返し実行します。