• ベストアンサー

C言語 型変換について

以下のように型変換すると a1 = 123.78、a2 = 123、b = 77、a3 = 78 となり、なぜb=77となってしまうのかわかりません。 詳しい方、すみませんが教えてください。よろしくお願いします。 float a1, a2, a3; int b; void main() { a1 = 123.78; a2 = (int)a1; b = (a1 - a2)*100; a3 = (a1 - a2)*100; printf("a1 = %3.3f, a2 = %3.3f, b = %3d, a3 = %3.3f" , a1, a2, b, a3); }

  • 科学
  • 回答数2
  • ありがとう数2

質問者が選んだベストアンサー

  • ベストアンサー
  • tadys
  • ベストアンサー率40% (856/2135)
回答No.2

カテ違いです。 C言語についての質問はこちらへ OKWave-> [技術者向] コンピューター ->プログラミング->C・C++ > a2 = (int)a1; a1 を int に直す時に小数点以下は切り捨てになるので a2 = 123.0 になります。 >b = (a1 - a2)*100; 最初に (a1 - a2) を計算します。 (a1 - a2) = 123.78 - 123.0 = 0.78 ですが、コンピュータの中では2進数で記憶されているので、0.78 を正確に表す事が出来ません。 せいぜい0.779999999......です。 これを100倍しても 77.9999999......なので、小数点以下を切り捨てると 77 になります。 78/100 を2進数で表すと 0.11000【11110101110000101000】 ただし、【】の部分を繰り返す循環小数です。

eggineer
質問者

お礼

丁寧な回答ありがとうございました。 二進数で0.78は表せれないんですね。盲点でした!

その他の回答 (1)

noname#212058
noname#212058
回答No.1

浮動小数点の『丸め誤差』のためです。浮動小数点の実際の 計算が、2進数で行われるために発生します。 google などで検索すると、解説ページがたくさん出てきますので、 詳しくはそちらで確認してください。

eggineer
質問者

お礼

丸め誤差をちゃんと理解していませんでした。 ありがとうございました。

関連するQ&A

  • c言語 型変換について

    c言語 型変換について 下記のように文字コードは、unsigned int型('B')をunsigned char 型(str[1] ) 格下げする型変換する規則を教えてください。 *質問ソースプログラム: int main(void) { char str[4]; /* 文字列を格納する配列 */ str[0] = 'A'; /* 代入 */ str[1] = 'B'; /* 代入 */ ・・・・・・ printf("size B %u\n",(unsigned)sizeof('B')); printf("size str[1] %u\n",(unsigned)sizeof(str[1])); * 実行結果 size B 4 size str[1] 1

  • C言語について

    C言語についてなのですが、error C4700: 初期化されていないローカル変数 'dright' が使用されますという表示がでてしまいます。今地下鉄の環状線の距離を計算して求めるというプログラミングを作っています。どうすれば直るのでしょうか。また、おかしいと思われる場所があれば教えていただきたいです。下がぼくが書いたソースです。 #define _CRT_SECURE_NO_WARNINGS #include <stdio.h> #include <stdlib.h> float d[15] = { 0.0, 0.0, 1.6, 3.0, 4.3, 5.4, 8.2, 8.9, 9.7, 14.1, 15.1, 17.2, 20.7, 22.2, 23.4 }; float dright(int a, int b)//右回りのときの計算 { if (d[a]<d[b]) { return d[b] - d[a]; } else { return d[a] - d[b]; } } int main(void) { int a, b; float dright, dleft; printf("出発駅番号の入力"); scanf("%d", &a); printf("到着駅番号の入力"); scanf("%d", &b); if (a = b)//出発駅と到着駅が同じ場合 { printf("出発駅と到着駅が同じです。もう一回やり直してください。"); } else { dleft = 26.4 - dright;//左回りの計算 if (dleft > dright)//左回りのほうの距離が長い場合 { printf("%f", dright);//右回りの距離を表示 } else { printf("%f", dleft); } } return 0; } どなたかお願いします。

  • C言語で・・

    実数データを入力し、小数点以下四捨五入したときの値を表示せよ。 自分では、以下のように作りました。 #include<stdio.h> int main(void) { float f_a,s; int i_a; printf("実数データを入力してください"); scanf("%f",&f_a); i_a=(int)f_a; s=f_a-i_a; if(s>=0.5){ i_a=i_a+1; } printf("%fを四捨五入すると%dです",f_a,i_a); return 0; } 他の作りかたがあれば、教えていただきたいのですが。IF文などの範囲でお願いします。

  • GCCで暗黙の型変換の警告を出したい

    情報が失われてしまうような代入について警告を出したいのですが、 どういったオプションを用いればよいでしょうか? コンパイラはGCCの3.x系か4.x系でお願いします。 以下のようなソースで型変換に関する警告がほしいんです。 --- test.c --- #include <stdio.h> int main(void) {   int a = 66000;   short b;   b = a; // <- 暗黙の型変換   printf("%d\n", b);   return 0; } 実行結果 $ ./test 464 以下のオプションを試しましたが、上記のソースでは 何の警告も出ませんでした。 -W -Wall -Wconversion -Wimplicit ご存知の方いらっしゃいましたら、どうかお助け下さい。

  • C言語の実数型の足し算

    C言語初心者です。関数の勉強していて、実数型計算に出くわしました。 #include <stdio.h> float add(float a, float b) { return a+b; } int main(void) { float x=10.5,y=20.3; printf("%f %f\n",x,y); printf("%f\n",add(x,y)); return 0; } としたら、 10.500000 20.299999 30.799999 という結果になりました。今のところint型でずーっと勉強していたので、20.3の20.299999表記が怪しく感じられ、結果も同様に怪しく感じられます。どうして、10.5+20.3=30.8とすっっきり表示してくれないのでしょうか。

  • C言語 プログラミングで行詰まりました…

    標準入力(キーボード)からi,jk,nの値を入力し、次の漸化式を計算し、X_0からX_nまで求めるプログラムを作成したいのですが、うまく表示されません。どかがおかしいのかご指摘お願いします。 <漸化式> X_n=(a+b)/X_(n-1) , X_0=c(n=0) ================================================================== #include<stdio.h> float f_X(int a,int b,float c) { float y; y=(a+b)/c; return y; } int main (void) { int number,i,j; float k,l,n,X; printf("i:"); scanf("%d", &i); printf("j:"); scanf("%d", &j); printf("k:"); scanf("%f", &k); printf("n:"); scanf("%f", &n); X=k; printf("X_0= %.6f\n",X); for(number=1;number<=n;number++) { l=f_X(i,j,X); printf("X_%d= %.6f \n",number,l); X=l; } return 0; } ===================================================================

  • c言語で

    c言語で /*a==bのとき商と余を求めるそれ以外は積を求める*/ # include<stdio.h> main () { int a,b,seki,sho,joyo; printf("aとbを入力してください\n"); scanf("%d %d", &a,&b); if(a==b) printf("sho=%d\n",(a/b)); printf("joyo=%d\n",(a%b)); else printf("seki=%d\n",(a*b)); return(0); } はどこがおかしいでしょうか? あと /*小数点*/ # include<stdio,h> main () { float a,b,seki,sho,joyo; printf("%f %f",&a,&b); if(a>b) printf("sho=%f4.3\n",(a/b)); printf("joyo=%f4.3\n",(a%b)); else printf("seki=%f4.3\n",(a*b)); return (0) にいたってはエラーの嵐なのですがどうすればいいでしょうか

  • C言語について

    「キーボードから10個の実数を入力し、それらの平均を求めるプログラムを作れ。」という問題で私は、 #include <stdio.h> void main(void) { float A; float a,b,c,d,e,f,g,h,i,j; printf("実数を入力してください:"); scanf("%f",&a); scanf("%f",&b); scanf("%f",&c); scanf("%f",&d); scanf("%f",&e); scanf("%f",&f); scanf("%f",&g); scanf("%f",&h); scanf("%f",&i); scanf("%f",&j); A=(a+b+c+d+e+f+g+h+i+j)/10; printf("平均は%fです.\n",A); } と考えたのですが、もっとすっきりとしたプログラムはないのでしょうか?もっとシンプルにしたいのです。教えて下さい。よろしくお願いします。

  • 型変換の問題?

    次のプログラムなんですが #include <stdio.h> #include <math.h> #include <cstdlib> #include <iostream.h> void main(){ double v[10000]; for(int b=0;b<10000;b++){ v[b]=2.5+0.01*(double)(rand()%249+1);} for(b=0;b<10000;b++){ int c=(int)(100.*v[b])-251; if(c==192){printf("%d %f\n",c,v[b]);} }} 私の環境のVC++6.0 on win2kのもとで やってみると、printf("%d %f\n",c,v[b])のところで 192 4.43 となったり 192 4.42 となったりします。 真の値は、192 4.43のつもりでプログラムを組んだのですが、 なぜかc=(int)(100.*v[b])-251;という写像は 単射になっていないようです。 さて、質問ですが、double→int型の型変換に原因があるのでしょうか?

  • C言語の簡単な質問です。

    #include "stdio.h" int tasizan(int a,int b){ int c; c=a+b; return c; } int main(void){ int a,b,c; a=1; b=1; c=tasizan(a,b); printf("%d+%d=%d",a,b,c); return 0; } と、あるとします。この文を読み込む順番を教えてください。  文の左に行番号みたいに順番を書き込んでください。   例 (例が間違ってたらすみません) 1 #include "stdio.h" 2 int main(void){ 3 int a,b,c; 4 a=1; 5 b=2; 6 c=a+b; 7 printf("%d+%d=%d",a,b,c); 8 return 0; }