gcc 编译多个源文件

一. 常用编译命令选项
假设源程序文件名为test.c。

1. 无选项编译链接
用法:#gcc test.c
作用:将test.c预处理、汇编、编译并链接形成可执行文件。这里未指定输出文件,默认输出为a.out。

2. 选项 -o
用法:#gcc test.c -o test
作用:将test.c预处理、汇编、编译并链接形成可执行文件test。-o选项用来指定输出文件的文件名。

3. 选项 -E
用法:#gcc -E test.c -o test.i
作用:将test.c预处理输出test.i文件。

4. 选项 -S
用法:#gcc -S test.i
作用:将预处理输出文件test.i汇编成test.s文件。

5. 选项 -c
用法:#gcc -c test.s
作用:将汇编输出文件test.s编译输出test.o文件。

6. 无选项链接
用法:#gcc test.o -o test
作用:将编译输出文件test.o链接成最终可执行文件test。

7. 选项-O
用法:#gcc -O1 test.c -o test
作用:使用编译优化级别1编译程序。级别为1~3,级别越大优化效果越好,但编译时间越长。

二. 多源文件的编译方法

如果有多个源文件,基本上有两种编译方法:
[假设有两个源文件为test.c和testfun.c]

1. 多个文件一起编译
用法:#gcc testfun.c test.c -o test
作用:将testfun.c和test.c分别编译后链接成test可执行文件。

2. 分别编译各个源文件,之后对编译后输出的目标文件链接。
用法:
#gcc -c testfun.c //将testfun.c编译成testfun.o
#gcc -c test.c //将test.c编译成test.o
#gcc -o testfun.o test.o -o test //将testfun.o和test.o链接成test

以上两种方法相比较,第一中方法编译时需要所有文件重新编译,而第二种方法可以只重新编译修改的文件,未修改的文件不用重新编译。

3. 如果要编译的文件都在同一个目录下,可以用通配符gcc *.c -o 来进行编译。

你是否会问,如果是一个项目的话,可能会有上百个文件,这样的编译法,人不是要累死在电脑前吗,或者等到你编译成功了,岂不是头发都白了,呵呵,所以我们要把上述的编译过程写进以下一个文本文件中:
Linux下称之为makefile
#这里可以写一些文件的说明
MyFirst: MyFirst.o hello.o
g++ MyFirst.o hello.o -o MyFirst
Hello.o:Hello.cpp
g++ -c Hello.cpp -o Hello.o
MyFirst.o:MyFirst.cpp
g++ -c MyFirst.cpp -o MyFirst.o
makefile 编写规则:
(1)以“#”开始的行为注释
(2)文件依赖关系为:
target:components
rule
存 盘为MyFirst,在终端输入:make MyFist,程序出现了错误可是所有程序员共同的敌人,在编写程序时我们应该尽量的去避免错误的出现,不过编写的时候再怎么都不可避免的出现这样那样的 错误,对程序 进行必要的调试是一个好主意,那我们怎么来调试程序呢,看下面:
gdb ./文件名 ////////////////在这里我修改下要想下面可以调试,在上面编译的 时候必须加上参数g,g++ -g hello.cpp -o hello
以下为调试状态下的可以用到的命令(可以仅输入单词的输入,如break可简为b),尖括号中为说明
list <显示源代码>
break 行号 <设置断点>
run <运行程序>
continue <继续从断点处执行>
print 变量 <调试时查看变量的值>
del 行号 <删除断点>
step <单步执行,可跟踪到函数内部>
next <单步执行,不可跟踪到函数内部>
quit <退出>
makefile 的编写不是件容易的事情,因为自己写的makefile可能不能在所有的unix/linux类操作系统下通用。因此在很多项目中都用automake.autoconf或者是Cmake等工具。


*******************
gcc格式
gcc [-c 编译,不连接] [-S] [-E] [-std] [-g 加入调试信息] [-pg加入剖析信息] [-Olevel]
    [-W 警告设置] [-pedantic 发出所有标准C列出的警告] [-Idir 添加包含目录到搜索列表中] [-Ldir添加库目录到搜索列表中]
    [-D] [-U] [-f] [-m] [-o 编译到可执行文件,并作一些优化]...
说明:
    -w       关闭所有警告设置
    -Wall    允许发出所有警告
    -werror  把所有警告转化为错误
    -v       显示编译过程中的每一步用到的命令
    -ggdb    在可执行程序中包含大量的调试信息
    -MM      输出一个make兼容的相关列表
    -O       优化编译过的代码
    -fsyntax-only  只检查语法,其他都不做
    -save-temps 保存编译过程中产生的中间文件
    -Q       编译器输出编译每个函数时所花的时间信息
gcc的优化选项:
    -O等价于-O1,并且包含线程直接跳转和迟延堆栈这两种优化。
    -O2包含了-O1的优化选项,并增加了一些额外的调整,如包含对处理器指令调度的调整。
    -O3也类似
gcc的调试选项:
    -g == -g2,该级别包括扩展符号表、行号、以及局部变量或外部变量。-g3除了包含-g2的信息外,还包含了源代码中定义的宏.
gcc的扩展:
    数据类型,如 long long来表示64位的存储单元。
    内联函数,inline关键字只是一个建议而不是指令,内联的优化由编译器分析完成。
    函数和变量的属性,关键字attribute,在函数或是变量的尾部加上_attribute_((属性名))来完成。
    void FunctionName() _attribute_((noreturn)); /* 不返回值 */
    int int_var _attribute((aligned 16)) = 0; /* 对齐到16位 */
    注释,//呵呵,这个在标准c中没有,标准c中的注释是/* */。
    构造函数名称,使用_FUNCTION_就可以输出当前调用函数的名称。如
    void main(void)
    {
         printf("%s", _FUNCTION_);
    }
【title】make的使用
    Makefile规则格式如下:
    target : dependency [dependency [...]]
             command
             command
             [...]
    target为目标体,即make要创建的文件;dependency依赖体,必须有依赖文件,即头文件和目标文件或源文件;
    警告:
    每个命令的第一字符必须为制表符,不能用8个空格来代替。
    Example:
    1.创建一个Makefile文件,用文本编辑
    2.输入下列代码
    helloworld : main.o function.o function.h
gcc main.o function.o -o helloworld
    main.o :  main.c function.h
gcc -c main.c
    function.o : function.c function.h
gcc -c function.c
    all : helloworld
    clean : /* 只是个动作 ,并不真的删除.o文件 */
rm *.o
    3.在目录中输入make就可以得到编译的文件了。
【error | make: **No targets specified and on makefile founcd. stop | 缺少Makefile文件名的文件】
【error | make: Circular function.o
    #include
    #include
    static char rcsid[] = "$Id$\n";
   
    int main(void)
    {
        extern char  **environ;
        char **my_env = environ;
        while (*my_env)
        {
            printf("%s\n", *my_env);
            my_env++;
         }
         exit(EXIT_SUCCESS);
     }
     用make prn_env.c编译后,在命令行输入$ident prn_env就可以得到$Id$的信息。
     用$rlog 文件名就可以从RCS数据库中检出该文件被修改的日志信息了。
     CVS的使用:
     1.使用$cvs -d dir init命令初始化cvs库。
     2.设置环境变量$CVSROOT,命令export CVSROOT = dir。
     3.导入目录到CVS中,先在CVSROOT下建立一个目录cdir,然后用命令cvs import [-m msg] cdir lpu2 start,执行就自动把当前目录中的文件导入到cvs/cdir下.

本文来自ChinaUnix博客,如果查看原文请点:http://blog.chinaunix.net/u1/51844/showart_450508.html

你可能感兴趣的:(gcc 编译多个源文件)