一、什么是隐式函数声明
在C语言中,库函数在调用前一般都要进行函数声明(include头文件)。如果没有声明,那么编译器会自动按照一种隐式声明的规则,为调用函数的C代码产生汇编代码。在有些情况下,隐式函数声明并不会对程序产生影响,但在有些情况下必须进行声明。
二、隐式函数声明会带来什么问题
2.1 隐式声明函数名称恰好在链接库中存在,但返回非int类型
前面给出的例子,并不会造成太大影响,因为在链接阶段很容易发现存在的问题。然而下面这个例子则会造成莫名的运行时错误。
#include <stdio.h>
int main(int argc, char** argv)
{
double x = sqrt(1);
printf("%lf", x);
return 0;
}
运行结果
1.000000
编译时会给出警告,提示隐式声明与内建函数’sqrt’不兼容。gcc编译器在编译时能够自动在常用库头文件(内建函数)中查找与隐式声明同名的函数,如果发现两者并不相同,则会按照内建函数的声明原型去生成调用代码。这往往也是程序员预期的想法。
上面的例子中隐式声明的函数原型为:
int sqrt(int);
而对应的同名内建函数原型为:
double sqrt(double);
最终编译器按照内建函数原型进行了编译,达到了预期效果。然而gcc编译器的这种行为并不是C语言的规范,并不是所有的编译器实现都有这样的功能。同样的源码在VC++2015下编译运行的结果却是:
VC++编译
warning C4013: “sqrt”未定义;假设外部返回 int
运行结果
2884223.000000
显然,VC++编译器没有没有所谓的“内建函数”,只是简单的按照隐式声明的原型,生成调用sqrt函数的代码。由于返回类型和参数类型的不同,导致错误的函数调用方式,产生莫名奇妙的运行时错误。
对着这种情况,由于返回类型的不同,两种编译器都可以给出警告信息,至少能引起程序员的注意。而下面这种情况,则更加隐蔽。
2.2 隐式声明函数名称恰好在链接库中存在,且返回int类型
测试代码如下:
#include <stdio.h>
int main(int argc, char** argv)
{
int x = abs(-1);
printf("%d", x);
return 0;
}
此时,由于隐式声明的函数原型与gcc的内建函数原型完全相同,所以gcc不会给出任何警告,结果也是正确的。
而VC++则仍然会给出警告:warning C4013: “abs”未定义;假设外部返回 int。
无论如何,隐式声明的函数原型与库函数完全相同,所以链接运行都是没有问题的。
下面,稍微改动一下代码:
#include <stdio.h>
int main(int argc, char** argv)
{
int x = abs(-1,2,3,4);
printf("%d", x);
return 0;
}
可见,gcc的内建函数机制并不关心函数的参数,只是关心函数的返回值。
vc++编译链接
warning C4013: “abs”未定义;假设外部返回 int
虽然这个例子的运行结果都是正确的,但是这种正确是“碰巧”的,因为额外的函数参数并没有影响到结果。这种偶然正确是程序中要避免的。
三、malloc隐式函数声明的解决方法
malloc作为C语言的库函数中的一个,其中头文件为#include<stdlib.h>/#include<malloc.h>,因此在使用malloc时,有两种方法。
1.包含头文件进行函数声明
#include<stdlib.h>
在使用malloc函数
2.不包含头文件使用
直接使用:__malloc_
例:a = (struct Node*)__malloc_(sizeof(struct Node));