首先C语言是一门面向过程的编程语言,它是由一系列的函数组成的。函数的使用必须遵守:
举个例子说明一下,下面是一个test.c的源程序:
#include
int main(){
sayHello();
return 0;
}
void sayHello(){
printf("Tom says:%s\n","Hello!");
}
我们在terminal上用gcc编译一下这个程序:
~$ gcc test.c -o test
然后就报了如下的错误:
test.c: In function ‘main’:
test.c:4:2: warning: implicit declaration of function ‘sayHello’; did you mean ‘ftello’? [-Wimplicit-function-declaration]
sayHello();
^~~~~~~~
ftello
test.c: At top level:
test.c:7:6: warning: conflicting types for ‘sayHello’
void sayHello(){
^~~~~~~~
test.c:4:2: note: previous implicit declaration of ‘sayHello’ was here
sayHello();
^~~~~~~~
正确的做法是在调用sayHello()函数前,先声明这个函数。这里解释一下,为什么在函数调用前需要先声明,这主要是由编译器的编译过程决定的:
test.c的完整且正确的程序如下。
下面这种是在源文件开头添加函数声明:
#include
void sayHello(); //函数声明
int main(){
sayHello();
return 0;
}
void sayHello(){
printf("Tom says:%s\n","Hello!");
}
或是下面这种调整main函数与sayHello函数的位置:
#include
void sayHello(){
printf("Tom says:%s\n","Hello!");
}
int main(){
sayHello();
return 0;
}
再次编译运行,就完全没有问题了:
~$ gcc test.c -o test
~$ ./test
Tom says:Hello!
补充:运行目标程序时要加上./,为什么呢?因为在类Unix系统上,必须指定可执行文件所在的目录,除非目录已添加到PATH环境变量中。./表示当前目录下。
其中调整函数的顺序其实是一件很痛苦的事情,尤其在加入新入的函数的时候,特别容易出现上述错误。如果写一些相互递归调用的函数,它们互相调用对方,那么总有一个函数在定义 前被调用了,那如何是好?有没有一种方法既不用调整代码,又能让编译器知道函数返回类型呢?有,将函数声明都写在源文件的开头或用头文件的形式将声明与定义进行分离。我们来说说用头文件的方式。
将函数声明放在一个.h的头文件中,然后再把.h文件包含进来就可以了。
函数声明只是一个函数签名,一条包含函数名、形参类型、返回类型的记录。我们将上述函数的声明放到一个叫tong.h的头文件中去:
void sayHello();
在test.c引入tong.h头文件
#include
#include "tong.h" //引入头文件
int main(){
sayHello();
return 0;
}
void sayHello(){
printf("Tom says:%s\n","Hello!");
}
这样一来,编译器在一开始就知道函数的返回类型,就不用稍后再找了,而且防止了编译器假设函数的返回类型,可以显式地告诉它了。这种告诉编译器函数会返回什么类型的语句就叫函数声明。如果代码中有很多函数,而你又不想管它们在文件中的顺序,那么可以在源文件开头列出函数声明,甚至可以把这些函数声明放到一个头文件中,再通过include指令包含进来。
小知识:
头文件的名称用双引号括起来和用尖括号括起来的区别:
1.用双引号括起来的文件名,编译器就会在本地查找文件,如果是加上了目录的文件名,编译器就会在相对路径下查找头文件;
2.用尖括号括起来的文件名,编译器就会在标准库里找,gcc知道在哪里。在类Unix系统中,头文件一般都放在/usr/local/include 、/usr/include这些地方。
例如我们要共享如下这个加密函数,那么我们需要想办法让其他程序知道它,这时就可以用头文件了:
void encrypt(char *message){
while(*message){
*message = *message ^ 31;
message++;
}
}
1.创建一个encrypt.h的头文件:
void encrypt(char *message);
2.在encrypt.c中包含这个头文件,这样可以让其他程序知道encrypt()函数:
#include "encrypt.h"
void encrypt(char *message){
while(*message){
*message = *message ^ 31;
message++;
}
}
3.在test1.c程序中使用这个函数,只需要包含这个头文件就可以了:
#include
#include "encrypt.h"
int main(){
char c[] = "hello world";
encrypt(c);
printf("%s\n",c);
return 0;
}
这样一来在主程序test1.c引入了encrypt.h,编译器就可以知道encrypt()函数,这样才能编译代码。最后,为了把所有东西编译到一起,只需要把源文件都传给gcc:
~$ gcc test1.c encrypt.c -o testEncrypt
~$ ./testEncrypt
wzssp?hpms{
这样把加密程序放到了一个单独文件中,就可以在任何程序中使用它了。