码迷,mamicode.com
首页 > 其他好文 > 详细

头文件和函数声明的另一个作用(转)

时间:2017-10-24 14:12:45      阅读:111      评论:0      收藏:0      [点我收藏+]

标签:编程   这不   运行时   调用   文件的   body   没有   复杂   col   

头文件的另一个作用,定义函数接口,作用似乎没那么大,因为编译、连接都通过了,程序也能运行了,这不就行了吗。下面我们用 一个例子说明这个问题。

假设我们写了一个很简单的程序: main调用了一个函数foo:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
#include <stdlib.h>
#include <stdio.h>
 
int main(void)
{
    int i;
 
    i = foo(2, 3);
    printf("foo returns %d\n", i);
 
    exit(0);
}
 
int foo(int a)
{
    return a + a;
}

此程序有严重的错误,但是如果我们用命令

1
$ gcc -c main.c

编译的时候,没有任何警告或出错信息。好,我们加上-Wall选项:

1
2
3
$ gcc -c -Wall main.c
main.c: In function `main‘:
main.c:8: warning: implicit declaration of function `foo‘

这句implicit declaration of function可能是被程序员忽视最多的警告了。 好,我们继续忽视它,接下来连接也能通过:

1
$ gcc -o ex1 main.o

运行也没有问题。 但你不觉得毛骨悚然吗? 一个严重的错误就这样从你眼皮底下过去了。你的程序越来越复杂,这个警告混在一大堆编译信息里,根本就注意不到了。 直到某一天一些奇怪的问题出现了,你开始调用各种土枪洋炮来调试程序…

其实,如果我们稍微尊重些编译器,把函数的声明加在main的前面,问题错误马上显现:

1
2
int foo (int a);
int main(void)

重新编译

1
2
3
$ gcc -c -Wall main.c
main.c: In function `main‘:
main.c:9: error: too many arguments to function `foo‘

这就是以错误的形式展现出来了,这就是函数声明的作用。 它既告诉程序员如何调用一个函数,也让编译器检查调用与函数原型是否一致。 有些人以为连接器会检查参数匹配的问题,连接不出错就万事大吉了,这是不对的。你想,参数是以寄存器或压栈的方式传递的。 编译之后,参数类型和个数等信息都已丢失,连接器还能帮你查错吗? 它只是简单地把名字相同的符号连接起来而已。

错误发现的越早越好

编程出现错误是不可避免的。错误发现的越早,修改的成本就越小。 因此原则是:尽量让错误暴露出来(例如严格的编译选项、测试),而不是掩盖或忽视它。能在编译时发现的错误,不要拖到运行时;能在编辑时发现的错误,不要拖到编译时(许多编辑器的括号匹配、代码补齐等功能就是为了减少这样的错误)。

 

头文件和函数声明的另一个作用(转)

标签:编程   这不   运行时   调用   文件的   body   没有   复杂   col   

原文地址:http://www.cnblogs.com/enyala/p/7722961.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!