作者duv (duv)
看板C_and_CPP
標題[語法] 在寫程式時 遇到將浮點數轉換成整數的問題
時間Tue Sep 15 13:11:35 2009
底下是一個我寫的簡單程式
程式是有關於計程車車資的計算問題:
假定計程車在1.5公里內, 車資都是70元;
超過1.5公里後, 每0.3公里, 加收5元 (不足0.3公里, 仍以0.3公里計)
請輸入搭乘里程數(以公里計)(可精確到小數點第三位,即可精確到公尺)(ex:1.853公里)
求算對應的車資為多少.
==============================================================================
底下則是我寫的程式
基本上程式可以跑出正確的結果
但由於搭乘里程數為浮點數
所以編譯器(我用dev C++這個編譯器)在編譯程式的時候
都會出現
In function 'int main()':
[Warning] converting to 'int' from 'double'
[Warning] converting to 'int' from 'double'
[Warning] converting to 'int' from 'double'
這樣的警示訊息
想請問 這樣的警示訊息 有可能是在警示什麼樣的負面情況呢@ @
又如果要避免出現這樣的警示訊息 我原來的程式寫法要怎麼改善(或改變)呢
==============================================================================
我寫的程式
float y;
int t;
t=(y-1.5)/0.3;
int z;
int w;
cout<<"請輸入里程(公里)[可輸入至小數點第三位的數字]";
cin>>y;
if (y<=1.500)
{
z=70;
}
else
{
if(0.3*(t+1.5)==y)
{
z=(y-1.5)/0.3;
w=70+5*z;
}
else
{
z=(y-1.5)/0.3;
w=70+5*(z+1);
}
}
cout<<"車資="<<w<<endl;
--
※ 發信站: 批踢踢實業坊(ptt.cc)
◆ From: 140.119.143.103
1F:→ NDark:強制轉型 z= (int) ( (y-1.5)/0.3; ) 該切掉就一定會被切掉 09/15 13:18
2F:→ NDark: z= (int) ( (y-1.5)/0.3 ) ; 09/15 13:19
3F:→ dendrobium:3.12 => 3 會遺失一些資訊 09/15 13:20
4F:推 holymars:顯式轉型就不會有warning 09/15 13:32
5F:→ MOONRAKER:為什麼不把金額設float?用轉型讓新手繞過問題,掩蓋他 09/15 14:55
6F:→ MOONRAKER:自己的規劃不良(又譯為亂寫),這樣真的好嗎? 09/15 14:56
7F:→ NDark:當然要先假設他宣告int是有理由的. 09/15 16:10
當然是有理由的!
如果今天某人搭乘距離是1.850公里
那1.850公里減掉基本運費的里程數(1.5公里)為
1.85-1.5=0.35(公里)
又因為每0.3公里加收五元(不足0.3公里以0.3公里計)
而(0.35/0.3)*5=1.1666666...
1.16666...*5= 5.83333333(元)
這樣會使得總運費只有70+5.83333333= 75.83333333
捨去小數得到的只是75元
根本就嚴重計算錯誤
正確算法應該是遇到
0.35/0.3=1.1666666...小數點後面有小數時
就要無條件進位讓1.1666666....變成2
這樣算出的總運費才會是正確的:70+5*2=80
這邊宣告t,z,w變數是整數 跟使用if選擇
就是為了達到可以發揮讓小數能無條件進位的功用
8F:→ NDark:再來 金額為什麼是浮點數? 也很值得深思 09/15 16:10
9F:→ MOONRAKER:有理由才怪哩 明明就是規劃不良 09/15 16:11
10F:→ MOONRAKER:還有金額是浮點數有什麼奇怪?你是在台灣才不需要用到 09/15 16:12
11F:→ MOONRAKER:小數金額 不要小數輸出的時候給個ceil或floor不就結了 09/15 16:13
才不是你所說的沒有理由!!(如我上面所說的 是為了要解決無條件進位的問題!)
你可以指點別人該怎麼寫會更好
但你劈頭就說別人是亂寫 還說別人那樣寫是沒有規劃 沒有理由
你這個人真的太失禮了!!!!!
你根本就沒有注意到這邊隱藏一個無條件進位的問題在
若不事先無條件進位
而是如你所說的到最後才把不要的小數金額捨去掉
根本就是錯誤的做法!!
12F:推 kingofsdtw:把最後結果 sscanf("%d%f",a,b)=> if b>=0.5 then a+1 09/15 18:35
13F:推 HOST1:我覺得可以用floor這個函數試試看喔 09/15 18:40
14F:→ HOST1:取整數然後+1 09/15 18:41
15F:→ sunneo:取到小數n位不就是 fround( x * 10^n ) / 10^n 嗎 = =... 09/15 21:40
16F:推 pttfly:請問一開始 t=(y-1.5)/0.3; y還沒給初值 好像沒意義 09/15 21:54
17F:→ pttfly:以至於後面的if沒意義 且if(0.3*(t+1.5)==y)中算式錯了 09/15 21:59
18F:→ duv:y是要使用者輸入的值 所以就不必給初值了... 09/15 22:27
19F:推 pttfly:所以 t=(y-1.5)/0.3; 要寫在 cin>>y; 之後 09/15 22:30
20F:→ duv:非常感謝指點m(_ _)m 09/15 22:54
21F:→ duv:然後我的if部分真的寫錯了orz 我應該要寫的是 09/15 23:04
22F:→ duv:if(0.3*t+1.5==y) orz 09/15 23:05
※ 編輯: duv 來自: 140.119.143.103 (09/15 23:06)
23F:推 pttfly:若你要這樣寫的話 t要改成用float喔 不能int 09/15 23:09
24F:推 VictorTom:關鍵是浮點數不要這樣直接用 == 比較才安全....Orz 09/15 23:52
25F:推 lance7:A這個人真的太失禮了!!!!! 09/17 11:39