近似与精确——《狂人C》习题解答15(第三章习题5)

814阅读 0评论2011-09-25 A13433758072
分类:

题目:从前有一个农夫,死后留下15头牛,他在遗书中写到:"妻子:分给全部牛的半数再加半头;长子:分给剩下的牛的半数再加半头;次子:分给剩下的牛的半数再加半头;长女:分给最后剩下的。"编程求长女得到了几头牛。
这是一个简单的小学算术问题:
15头牛的一半是7又1/2 ,再加半头得8,这是妻子所得。剩下7头
7头牛的一半是3又1/2 ,再加半头得4,这是长子所得。剩下3头
3头牛的一半是1又1/2 ,再加半头得2,这是次子所得。剩下1头
因而长女所得为1头。
然而如果写出如下的代码,则最多只能得60分。
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #define ZONGSHU 15. //总数:留下15头牛
  4. #define FENPEI_BL .5 //分配比例: 半数
  5. #define EWAI_TJ .5 //额外添加:半头
  6. int main( void )
  7. {
  8.   double qizi , zhangzi , cizi , zhangnv ; //妻子、长子、次子、长女所得
  9.   double shengyu = ZONGSHU ; //剩余的数量
  10.   
  11.   qizi = shengyu * FENPEI_BL + EWAI_TJ ; //妻子所得
  12.   shengyu -= qizi ; //剩余的数量
  13.   
  14.   zhangzi = shengyu * FENPEI_BL + EWAI_TJ ; //长子所得
  15.   shengyu -= zhangzi ; //剩余的数量
  16.   
  17.   cizi = shengyu * FENPEI_BL + EWAI_TJ ; //次子所得
  18.   shengyu -= cizi ; //剩余的数量
  19.   
  20.   zhangnv = shengyu ; //长女:分给最后剩下的
  21.   printf("长女得到了%f头牛\n" , zhangnv ) ;
  22.          
  23.   system("PAUSE");
  24.   return 0;
  25. }
输出:长女得到了1.000000头牛
因为,第一,这个结果仅仅表示长女得到的牛数约等于1头;第二,代码并没有真正实现前面的算术运算过程。譬如
qizi =  shengyu * FENPEI_BL +  EWAI_TJ ;
所表示的含义仅仅是一些近似的值的一个近似运算,而非前面算术运算过程中的精确运算。因为就其本质和普遍情形来讲,实浮点类型的数据只是对实数的一个近似表示,这注定实浮点类型的运算也只是一种近似运算。只不过在本题目中,近似的精度很高,计算结果恰好和精确的结果一致而已。如果把程序视为对笔算过程的精确模拟的话,显然前面一段代码并不符合要求。
在计算机中,只有整数类型是对整数集合子集的近似表示。所以如果希望准确地模拟笔算过程就只能用整数类型。然而笔算过程涉及到了分数。在数学中,分数也是一种精确表示,然而在C语言中却并没有与之对应的“分数类型”。
没有相应的数据类型怎么办?答案很简单:没有这种类型就创造这种数据类型。为创造性提供了广阔的发挥空间是C语言的特点和魅力,也恰恰是编程的乐趣之一。
由于分数是由分子、分母两个部分组成,而分子、分母都是整数,因而可以用两个整数类型的数据来表示分数。对于这样的数据,C语言并没有提供直接的运算,这种“分数”的运算需要自己用C语言所提供的运算模拟。
例如,若计算a/b+c/d,则无法通过一次“+”运算完成,只能分两次计算出和的分子“b*c+d*c”及和的分母“a*c”。
按照这种方法得到的代码是
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #define ZONGSHU_FZ 15 //总数的分子
  4. #define ZONGSHU_FM 1 //总数的分母
  5. #define FENPEI_BL_FZ 1 //分配比例的分子
  6. #define FENPEI_BL_FM 2 //分配比例的分母
  7. #define EWAI_TJ_FZ 1 //额外添加的分子
  8. #define EWAI_TJ_FM 2 //额外添加的分母
  9. int main( void )
  10. {
  11.   int qizi_fm , qizi_fz , //妻子所得的分母和分子
  12.        zhangzi_fm , zhangzi_fz , //长子所得的分母和分子
  13.        cizi_fm , cizi_fz , //次子所得的分母和分子
  14.        zhangnv_fm , zhangnv_fz ; //长女所得的分母和分子
  15.   int shengyu_fm = ZONGSHU_FM , //剩余的数量的分母
  16.        shengyu_fz = ZONGSHU_FZ ; //剩余的数量的分子
  17.   
  18.   qizi_fz = shengyu_fz * EWAI_TJ_FM //妻子所得
  19.            + shengyu_fm * FENPEI_BL_FM * FENPEI_BL_FZ;
  20.   qizi_fm = shengyu_fm * FENPEI_BL_FM * EWAI_TJ_FM ;
  21.   shengyu_fz = shengyu_fz * qizi_fm - qizi_fz * shengyu_fm ; //剩余的数量
  22.   shengyu_fm *= qizi_fm ;
  23.   
  24.   zhangzi_fz = shengyu_fz * EWAI_TJ_FM //长子所得
  25.               + shengyu_fm * FENPEI_BL_FM * FENPEI_BL_FZ;
  26.   zhangzi_fm = shengyu_fm * FENPEI_BL_FM * EWAI_TJ_FM ;
  27.   shengyu_fz = shengyu_fz * zhangzi_fm - zhangzi_fz * shengyu_fm ; //剩余的数量
  28.   shengyu_fm *= zhangzi_fm ;
  29.        
  30.   cizi_fz = shengyu_fz * EWAI_TJ_FM //次子所得
  31.            + shengyu_fm * FENPEI_BL_FM * FENPEI_BL_FZ;
  32.   cizi_fm = shengyu_fm * FENPEI_BL_FM * EWAI_TJ_FM ;
  33.   shengyu_fz = shengyu_fz * cizi_fm - cizi_fz * shengyu_fm ; //剩余的数量
  34.   shengyu_fm *= cizi_fm ;
  35.   
  36.   zhangnv_fz = shengyu_fz ; //长女所得
  37.   zhangnv_fm = shengyu_fm ;
  38.   printf("长女得到了%d又%d/%d头牛\n" ,
  39.           shengyu_fz/shengyu_fm , shengyu_fz % shengyu_fm , shengyu_fm ) ;
  40.   system("PAUSE");
  41.   return 0;
  42. }
输出:长女得到了1又0/16384头牛
这是一个精确的结果。
【注:学习了控制语句和函数理论之后,后一个代码可以进一步改进。】
 
上一篇:牛人博客/网站
下一篇:数组与指针概念剖析