C语言辨析——深入理解格式字符的用法

1. 问题

下面程序为什么的输出结果为什么不是25而是0?问题出在哪?


#include <stdio.h>
#include <math.h> 
int main()
{
  int a=3,b=4;  
  printf("%d\n",pow(a,2)+pow(b,2));    
  return 0;
}

2. 分析

函数pow的返回类型是double,因此表达式pow(a,2)+pow(b,2)的类型是double。而printf函数中的输出格式字符是“%d”,其对应的表达式应该是int类型。目前我们所用计算机系统大多数以8字节存储double类型数据,以4字节存储int类型数据,因此,可以得出结论:由于输出格式字符与表达式类型不匹配而产生错误输出。

3. 改正

(1)修改输出格式字符为“%f"或“%lf"。

(2)利用强制转换类型,将double类型的表达式转换为int类型。

修改后的程序如下所示。


#include <stdio.h>
#include <math.h> 
int main()
{
  int a=3,b=4;  
  printf("%d\n",pow(a,2)+pow(b,2)); // error    
  printf("%f\n",pow(a,2)+pow(b,2)); // (1)    
  printf("%lf\n",pow(a,2)+pow(b,2));// (1)
  printf("%d\n",(int)(pow(a,2)+pow(b,2)));// (2)
  return 0;
}

4. 进一步分析:为什么错误输出的结果是0

我们来分析表达式pow(a,2)+pow(b,2)的存储值,下面程序可以输出该表达式所占8个字节的值。

#include <stdio.h>
#include <math.h> 
int main()
{
  int a=3,b=4;    
  double c = pow(a,2)+pow(b,2);
  char *p = (char *)&c;
  for(int i=0; i<sizeof(double); i++) {
    printf("%02x ",*(p+i));
  }
  return 0;
}

输出结果是:

00 00 00 00 00 00 39 40

在错误的程序中,输出格式字符"%d"对应前4个字节,它们的值都是00,因此,输出结果为0。如果再输出一个"%d"会是多少呢?应该是0x40390000,即十进制的1077477376。我们通过程序来验证。


#include <stdio.h>
#include <math.h> 
int main()
{
  int a=3,b=4;    
  printf("%d\t%d\n",pow(a,2)+pow(b,3));
  return 0;
}

图片

注:以上结论是在Dev C++ 5.11下调试的结果,在其他编译器下是什么结果就不知道了。在大端系统中,会是什么情况呢?下面以截图展示。

图片

图 大端序存储

图片

图 运行结果是个随机值

与小端序有很大不同,

可见用错格式字符,问题很严重!

图片

参考文献:

[1]李红卫,李秉璋. C程序设计与训练(第四版)[M],大连,大连理工大学出版社,2023.

[2]https://pan.baidu.com/s/17ZXphwqySNIsIgcGtYMjvg?pwd=lhwc

  • 10
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

lhw---9999

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值