GCC编译器(2)
4、警告选项
在编译过程中,编译器的报错和警告信息对于程序员来说是非常重要的信息,GCC包含完整的出错检查和警告提示功能,它可以帮助Linux程序员尽快找出错误的或潜在的错误代码,从而写过更优美的代码。GCC的编译器警告选项如下表:
类型 | 说明 |
-Wall | 启用所有警告信息 |
-Werror | 在发生警告时取消编译操作,即将警告看作是错误 |
-w | 禁用所有警告信息 |
下面看一段代码,使用GCC编译,同时开启警告信息:
#include <stdio.h> void main () { int x; for(x=1;x<=10;x++) { printf("%d\n",x); } }
对上面的代码进行编译连接:
$ gcc -Wall example3.c -o example3
example3.c:2:6: 警告: ‘main’的返回类型不是‘int’ [-Wmain]
从上面的输出看到,GCC给出了警告信息,意思是main函数的返回值被声明为void,但实际应该是int。
此外,GCC还提供了许多以-W开头的选项,允许用户指定输出某个特定的警告,例如:
- -Wcomment:出现注释嵌套时发出警告。
- -Wconversion:如果程序中存在隐式类型转换,则发出警告。
- -Wformat:检查printf和scanf等格式化输入输出函数的格式字符串和参数类型的匹配情况,如果发现不匹配则发出警告。
- -Winline:如果函数不能被内联,则发出警告。
- -Wlong-long:如果使用了long long型数据,则发出警告。
- -Wmain:如果main函数的返回类型不是int型,或者调用main函数时使用的参数数目不正确,则发出警告。
- -Wmissing-declarations:如果定义了全局函数,但却没有在头文件中声明,则发出警告。
- -Wparentheses:在某些情况下,如果忽略掉了括号,则会发出警告。
- -Wreturn-type:如果函数定义了返回类型,而默认类型是int型,编译器会发出警告。
- -Wuninitialized:如果使用的自动变量没有被初始化,则发出警告。
- -Wundef:如果在#if宏中使用了未定义的变量做判断,则发出警告。
- -Wunused:如果声明的变量或static型函数没有使用,则发出警告。
下面使用GCC编译一段程序,来说明开启警告信息的必要性:
#include<stdio> int main() { double x; printf("%d\n",x); /* 这里将%f误输为%d */ return 0; }
对上面的程序进行编译:
$ gcc example4.c -o example4
可以看到,编译并没有报错,运行可执行文件,输出结果为:
$ ./example4
134513689
这不是想要的输出结果,如果在上面的编译中加入-Wformat或-Wall选项,即:
$ gcc -Wformat example4.c -o example4
或
$ gcc -Wall example4.c -o example4
GCC给出如下警告信息:
example4.c: 在函数‘main’中:
example4.c:5:5: 警告: 格式 ‘%d’ expects argument of type ‘int’, but argument 2 has type ‘double’ [-Wformat]
格式字符串和参数类型的不匹配会导致程序运行错误,所以这是是非常有用的警告选项。
下面使用GCC编译一段程序,使用-Wparentheses选项对其中的括号进行检查。
#include<stdio> int main() { int a=1; int b=0; int c=1; if(a&&b||c) { ; } if(a==1) if(b==1) printf("b=1\n"); else printf("b!=1\n"); return 0; }
对上面的程序进行编译:
$ gcc -Wparentheses example5.c -o example5
example5.c: 在函数‘main’中:
example5.c:7:5: 警告: 建议在‘||’的操作数中出现的‘&&’前后加上括号 [-Wparentheses]
example5.c:11:7: 警告: 建议显式地使用花括号以避免出现有歧义的‘else’ [-Wparentheses]
所以GCC编译器的警告选项对程序员来说是非常重要的。
5、连接选项
GCC编译器提供的连接器选项如下表:
类型 | 说明 |
-Idirectory | 向GCC的头文件搜索路径中添加新的目录 |
-Ldirectory | 向GCC的库文件搜索路径中添加新的目录 |
-llibrary | 提示连接程序在创建可执行文件时包含指定的库文件 |
-static | 强制使用静态链接库 |
-shared | 生成动态库文件 |
先来理解一下头文件和库文件这两个概念:
头文件包含变量和函数的声明,但没有定义函数的实现。函数的具体实现实在库文件中完成的,库文件可分为静态库和动态库,静态库是指编译连接时,将库文件的代码全部加入到可执行文件中,这样运行时就不需要库文件了。静态库的后缀名一般为“.a”。动态库是指在编译连接时并不将库文件的代码加入到可执行文件中,而是在程序执行时由运行时连接文件加载库文件,这样可以节省系统的开销。动态库的后缀名一般为“.so”。
例如我们编译是用-I选项来指定头文件的路径:
$ gcc example.c –o example –I/home/xxx/include
头文件所对应的库文件,如果没有特别指定时,GCC会到默认的搜索路径进行查找。
使用-L选项来指定库文件的路径,例如:
$ gcc example.c –o example –L/home/xxx/lib
GCC编译器在默认情况下使用动态库,但如果使用了-static选项,连接器将忽略动态库,强制使用静态链接库,即使用如下命令:
$ gcc example.c –o example –static –lm
此时静态库文件中的代码全部包含到可执行文件中,所以生成的可执行文件比较大。
本节完…