本文是C编程中一些常见错误的总结,有些是显而易见的,有些则是不容易发现

本文地址:http://www.cnblogs.com/archimedes/p/get-screwed-c.html,转载请注明源地址。

1.忘记注释结束符

代码

a = b; /* 出现bug
c = d; /* c = d将不会执行*/

2.意外的赋值/意外的bool表达式

代码:

if(a = b) c;  /*a恒等于b,只有当b!=0的时候才执行*/

再看下面的代码:

if( < a < ) c;   /*布尔表达式恒为真*/

上面代码中的bool表达式恒为真,由于0 < a的值为0或1,永远都小于5成立,当然C中没有bool表达式,这里只是假设

3.有缺陷的宏

代码:

#define assign(a,b) a=(char)b
assign(x,y>>)

变成

x=(char)y>>    /* 可能不是你的目的 */ 

4.不匹配的头文件

代码:

//foo.h:
struct foo { BOOL a};
//F1.c
#define BOOL char
#include "foo.h"
//F2.c
#define BOOL int
#include "foo.h"

F1与F2对结构体foo中的BOOL类型定义不一致,如果交互就会出现错误

5.飘忽不定的返回值

可能你会写下面的代码:

int foo (a)
{ if (a) return(); } /* bug,因为有时候没有值返回 */

6.不可预知的结构体

看下面的比特包结构:

struct eeh_type
{
uint16 size: ; /* 10 bits */
uint16 code: ; /* 6 bits */
};

取决于用哪个C编译器,还有你的机器使用大小端,这段代码实现为:

<10-bits><6-bits> 或 <6-bits><10-bits>

同时取决于C编译器、机器体系结构、不可思议的优先设置,这些项可能对齐到最近的8, 16, 32, or 64 bits.

7.不确定的求值顺序

  foo(pointer->member, pointer = &buffer[]);

不同的编译器针对函数参数有不同的求值顺序,gcc是从左到右的求值顺序,有的编译器是从右往左

8.容易改变的块作用域

代码:

if( ... )
foo();
else
bar();

当加上调试输出信息:

if( ... )
foo();
else
printf( "Calling bar()" ); /* 注意! else止于此 */
bar(); /* 注意! bar永远会被执行 */

9.不安全的返回值

代码:

char *f() {
char result[];
sprintf(result,"anything will do");
return(result); /* 注意! result 被分配在栈上. */
} int g()
{
char *p;
p = f();
printf("f() returns: %s\n",p);
}

10.未定义副作用(side effects

哪怕一个简单的表达式,C没有定义副作用的顺序,结果取决于你的编译器,i/i++可能等于0或1,看下面的代码:

#include <stdio .h>
int foo(int n) {printf("Foo got %d\n", n); return();}
int bar(int n) {printf("Bar got %d\n", n); return();}
int main(int argc, char *argv[])
{
int m = ;
int (*(fun_array[]))();
int i = ;
int ii = i/++i;
printf("i/++i = %d, ",ii);
fun_array[] = foo; fun_array[] = bar;
(fun_array[++m])(++m);
return ;
}

11.未初始化的局部变量

事实上这个bug不是那么出名,但是一旦发生严重性不会输给其他的bug,看下面的代码:

void foo(a)
{ int b;
if(b) {/* bug! b没有被初始化 */ }
}

现代编译器会发出错误的警告,看下面代码:

void foo(int a)
{ BYTE *B;
if(a) B=Malloc(a);
if(B) { /* BUG! B 可能没被初始化 */ *b=a; }
}

12.杂乱的编译时环境

编译时环境产生成百上千的编译信息,我们对此知之甚少。有些危险的常用名,导致很难被发现:

#include <stdio.h>
#define BUFFSIZE 2048
long foo[BUFSIZ]; //注意拼写: BUFSIZ != BUFFSIZE

编译器不会报错,因为BUFSIZ已经在stdio.h中定义

13.八进制数

在C中,八进制数以0开始,如果数字中没有‘8’或‘9’出现,编译器不会警告

 int numbers[] = { ,
, // 8 而不是 10
014 }; // 12, 而不是 14

14.有符号字符

C中的有符号char可能出现各种错误,比如128是一个负数,另外,任何使用低精度整数都必须十分的小心,C使得这些太容易被忽略了

char s = ;
unsigned char u = ;
s++; /* 结果是负数 */
if (s<u) { /* true!*/ }
if(s>) { /* 永不可能为true */ }
if(u<) { /* 永不可能为true*/ }

15.糟糕的“标准库”

代码:

{ int a=,b=;
char buf[];
scanf("%d %d",a,b); // &a,&b?
sprintf(buf,"this is the result: %d %d"); // 溢出
}

C编程常见问题总结的更多相关文章

  1. android编程常见问题-程序真机中不显示

    新手编程常见问题: 问题表现:连接上手机后,程序不显示 解决版本:检查AndroidManifest.xml 文件中SDK版本的设置(要求要兼容当前手机版本系统),如下:

  2. android编程常见问题-程序在模拟器中不显示

    新手编程常见问题: 问题表现:程序运行成功,但是在模拟器中不显示 解决办法:检查项目版本和模拟器版本是否匹配或兼容,如果不匹配,选择和模拟器版本一致 项目版本:右键-Properties-androi ...

  3. Python 编程常见问题

    Python 编程常见问题 经常使用Python编程,把经常遇到问题在这里记录一下,省得到网上查找,因此这篇文章会持续更新,需要的可以Mark一下.进入正题: 1.Python常用的文件头声明 #!/ ...

  4. 【Networking】网络编程常见问题汇总

    [百度分享]网络编程常见问题总结 串讲(一)网络编程常见问题总结    在网络程序中遇到的一些问题进行了总结, 这里主要针对的是我们常用的TCP socket相关的总结, 可能会存在错误, 有任何问题 ...

  5. 基于RequireJS和JQuery的模块化编程——常见问题解析

    由于js的代码逻辑越来越重,一个js文件可能会有上千行,十分不利于开发与维护.最近正在把逻辑很重的js拆分成模块,在一顿纠结是使用requirejs还是seajs的时候,最终还是偏向于requirej ...

  6. android编程常见问题-No Launcher activity found!

    新手编程常见的问题: 问题表现: console提示:No Launcher activity found! The launch will only sync the application pac ...

  7. Java编程常见问题汇总

    每天在写Java程序,其实里面有一些细节大家可能没怎么注意,这不,有人总结了一个我们编程中常见的问题.虽然一般没有什么大问题,但是最好别这样做.另外这里提到的很多问题其实可以通过Findbugs( h ...

  8. IO模式设置网络编程常见问题总结—IO模式设置,阻塞与非阻塞的比较,recv参数对性能的影响—O_NONBLOCK(open使用)、IPC_NOWAIT(msgrcv)、MSG_DONTWAIT(re

    非阻塞IO 和阻塞IO: 在网络编程中对于一个网络句柄会遇到阻塞IO 和非阻塞IO 的概念, 这里对于这两种socket 先做一下说明:       基本概念: 阻塞IO:: socket 的阻塞模式 ...

  9. CUDA编程常见问题 转

    http://blog.csdn.net/yutianzuijin/article/details/8147912 分类: 编程语言2012-11-05 10:55 2521人阅读 评论(0) 收藏  ...

随机推荐

  1. 【软件分析与挖掘】BOAT: An Experimental Platform for Researchers to Comparatively and Reproducibly Evaluate Bug Localization Techniques

    摘要: 目前有许多的bug定位技术,但是,由于他们基于不同的数据集,而且有些数据集还不是公开的,甚至有些技术只应用于小数据集,不具有通用性,因此,不好比较这些技术之间的优劣. 因此,BOAT应运而生. ...

  2. Qt5 从头学(1)-- 环境

    对我来说MFC太过麻烦了,同样是桌面开发工具,Qt就完全不一样了.Qt使用C++语言可以轻松实现"一次编写,到处编译"的跨平台性能,并且可以做出很多炫酷的界面效果.目前支持几乎所有 ...

  3. Rpath handling on Linux

    The solution in the article below seems promising: http://www.blaenkdenum.com/notes/cmake/#rpath set ...

  4. 修改efi分区

    换了ssd硬盘,速度真快! 我特意准备了一个efi分区(fat32,忘记指定efi了),然而win10安装时不能指定启动位置,自己创建了一块. 下面我们把让win10使用自己创建的efi分区. 以管理 ...

  5. NavMesh系统动态碰撞的探讨

    Unity3D提供的NavMesh系统可以方便的解决游戏的寻路问题,但是该系统有一个比较让人不理解的问题: NavMesh导航时会忽略Physics系统本身的碰撞,也就是说NavMeshAgent在移 ...

  6. AssetBundle系列——场景资源之打包(一)

    本篇讲解的是3D游戏的场景资源打包方式,首先简单的分析一下场景中所包含的资源的类型. 场景资源一般包含:地表模型(或者是Unity Terrain),非实例化物体(摄像机.空气墙.光源.各种逻辑物体之 ...

  7. WCF小白初试 错误之一:“有零个应用程序终结点”的解决办法

    遇到这类问题 应该是配置文件出现了问题 解决办法是将配置文件中的<service name="命名空间+类名">就可以解决

  8. bootstrap插件学习-bootstrap.popover.js

    先看bootstrap.popover.js的结构 var Popover = function ( element, options ){} //构造器 Popover.prototype = {} ...

  9. SQL Server 2014里的IO资源调控器

    在本文中,我们将来看看SQL Server 2014在资源调控器方面增加了哪些新的功能.资源调控器(Resource Governor)是从SQL Server 2008开始出现的一项功能.它是用于管 ...

  10. Javascript之旅(一)

    Javascript之旅(一) 一.基础知识 基本语法 变量 数据类型 字符串 数组 对象 条件判断 循环 Map和Set iterable 为什么要学习JavaScript JavaScript 是 ...