亲历fread函数的错误用法及修改验证过程------台上1分钟, 台下1天功

        我曾经在http://blog.csdn.net/stpeace/article/details/42847697中说过:fwrite和fread函数针对的是字符(无边界), 而不是字符串(以'\0'作为边界)


        今天呢, 某模块出了一个问题。 据说, 这个问题搞了较长时间。我准备晚上去打球的,毕竟感觉好久没有打了尴尬。 结果一个电话过来, 说有个紧急问题需要我协助定位, 我就乖乖返回了。经了解问题现象并查看源代码, 我立即发现了fread函数用法错误。后来修改并验证, 果然如此。

        fread函数错误用法的程序抽象为:


 
 
  1. #include <iostream>
  2. using namespace std;
  3. typedef struct _node
  4. {
  5. char name[ 100];
  6. int score;
  7. }Student;
  8. int main()
  9. {
  10. Student s;
  11. memset(&s, 0, sizeof(s));
  12. strncpy(s.name, "Eric", sizeof(s.name) - 1);
  13. s.score = 99;
  14. FILE *fp = fopen( "taoge", "wb");
  15. fwrite(&s, sizeof(s), 1, fp);
  16. fwrite(&s, sizeof(s), 1, fp);
  17. fwrite(&s, sizeof(s), 1, fp);
  18. fclose(fp);
  19. Student t;
  20. memset(&t, 0, sizeof(t));
  21. fp = fopen( "taoge", "rb");
  22. fread(&t, sizeof(t) - 1, 1, fp); // danger, logical error
  23. cout << t.name << endl;
  24. cout << t.score << endl;
  25. memset(&t, 0, sizeof(t));
  26. fread(&t, sizeof(t) - 1, 1, fp); // danger, logical error
  27. cout << t.name << endl;
  28. cout << t.score << endl;
  29. memset(&t, 0, sizeof(t));
  30. fread(&t, sizeof(t) - 1, 1, fp); // danger, logical error
  31. cout << t.name << endl;
  32. cout << t.score << endl;
  33. fclose(fp);
  34. return 0;
  35. }
       结果为:

Eric
99


25344


6488064

      我们看到, fread少读了一个字符, 结果会导致后面读取的时候出现整体偏移错误。 应该修改为:


 
 
  1. #include <iostream>
  2. using namespace std;
  3. typedef struct _node
  4. {
  5. char name[ 100];
  6. int score;
  7. }Student;
  8. int main()
  9. {
  10. Student s;
  11. memset(&s, 0, sizeof(s));
  12. strncpy(s.name, "Eric", sizeof(s.name) - 1);
  13. s.score = 99;
  14. FILE *fp = fopen( "taoge", "wb");
  15. fwrite(&s, sizeof(s), 1, fp);
  16. fwrite(&s, sizeof(s), 1, fp);
  17. fwrite(&s, sizeof(s), 1, fp);
  18. fclose(fp);
  19. Student t;
  20. memset(&t, 0, sizeof(t));
  21. fp = fopen( "taoge", "rb");
  22. fread(&t, sizeof(t), 1, fp); // ok
  23. cout << t.name << endl;
  24. cout << t.score << endl;
  25. memset(&t, 0, sizeof(t));
  26. fread(&t, sizeof(t), 1, fp); // ok
  27. cout << t.name << endl;
  28. cout << t.score << endl;
  29. memset(&t, 0, sizeof(t));
  30. fread(&t, sizeof(t), 1, fp); // ok
  31. cout << t.name << endl;
  32. cout << t.score << endl;
  33. fclose(fp);
  34. return 0;
  35. }
     结果:

Eric
99
Eric
99
Eric
99

      正常。


      为什么写代码的人当时要用sizeof(...) - 1呢? 

      根本原因: 没有搞清楚fread函数的用法。在这里, 我再次强调一遍: fread, read, recv, fwrite, write, send这类函数针对的是字符(无边界), 而不是字符串(以'\0'作为边界), 与'\0'没有半毛钱的关系。

      间接原因: 我们经常强调数组szTest必须以'\0'结尾, 可能是当时写该代码的人受了这个思想的影响, 以为任何缓冲区的最后一定是'\0'.


       我以前总结过类似问题, 所以今天很快帮助其他同事解决了此问题, 好开心大笑, 尽管最后球没打成。 


       在其位, 谋其事, 体现自身价值微笑



       台上1分钟, 台下1天功啊, 以后继续坚持学习、实践、总结、分享。  好吧, 本文就总结到这里。






  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值