##gcc编译过程
预处理,编译,汇编,链接,反汇编
一般用的语句:
gcc -c -o xxx.c xxx.o
当.c中出现语法错误时,报错的环节在编译.
-c表示只编译(compile)源文件但不链接,会把.c或.cc的c源程序编译成目标文件,一般是.o文件。
-o用于指定输出(out)文件名。不用-o的话,一般会在当前文件夹下生成默认的a.out文件作为可执行程序。
小常识:
.c中引入头文件的<>和""
#include <stdio.h>
是从系统目录中寻找头文件并引入;
(即工具链所在的位置,一般时系统默认位置)
#include “stdio.h”
是从项目所在的位置寻找头文件并引入;
当提示寻找不到头文件时,可以注意自己的引入格式是否正确,如果还提示查找不到,可以在语句后面加上-I ./目录
即人为指定目录;又或者可以将头文件放到系统目录中去.
gcc -c -o main.o main.c -I ./
当.o文件很多时,可以构建库,分为静态库和动态库;
链接时;
gcc -o text main.o libsub.a
(链接静态库是 libxxx.a)
将静态库和main.o链接在一起生成一个名字为text的应用,一般链接静态库比动态库大得多.
gcc -o text main.o libsub.so
(链接动态库是 libxxx.so)
不过一般使用的是-lxxx 这样就能省略前缀名lib和后缀.a或者.so
gcc -o text main.o -lsub
不过一般会报错找不到库文件,原理和头文件一致,你可以把生成的库文件放到系统目录
也可以在命令中指定位置(使用-L ./)
gcc -o text main.o -L ./ -lsub
执行应用时静态库没问题,但执行动态库生成的应用会报错,因为链接过程和执行过程是分开的,执行时使用的时默认路径,默认路径中没有自己生成的库文件,所以会报错,你可以把库文件放进库文件,也可以把库文件位置添加进默认路径中(像添加系统变量一样).
export LD_LIBRARY_PATH = $LD_LIBRARY_PATH:/目录
##Makefile引入与规则
gcc操作的缺点:gcc会把语句内的.c文件按顺序一一处理,当文件很多的时候,编译一次没问题,但当我只修改了一个.c文件其他没修改时,再编译一次gcc就会把所有文件都编译一遍,这样带来的效率会变低,应该把.c文件都一条一条的分别执行.
而Makefile就能实现这一条条gcc语句的执行;
gcc -o a.o a.c
gcc -o b.o b.c
gcc -o text a.o b.o
在Makefile中,他会基于时间来判断哪个文件需要重新编译,在第一条中如果a.o比a.c新(创建时间比a.c迟)那么a.o就不用编译(a.c没有被修改)
如果b.c比b.o新,那就意味着b.c被修改了,那么b.o就需要重新编译.
在最后一条中,如果a.o和b.o都存在,且任一.o文件比text新的话,就说明有文件更新了,需要重新生成text.
这就是Makefile的流程,而Makefile的语法如下:
注意命令行前面是用回车缩进,空格不可以存在,否则会报错
下面是正常Makefile的样子:
后面还有Makefile的变量,通配符,赋值等操作需要了解.
在命令中变量要用$()括起来
通配符%.o
$@ 表示目标
$< 表示集合中第一个子集或元素
$^ 表示整个集合
*.o 表示shell的所有.o文件
.PHONY 假想目标(目标冲突时使用)
即时变量,延迟变量
A := xxx #即时变量,赋值时即可确定
B = xxx #延时变量,使用到时才确定
?= #如果时第一次定义才起效,如果在前面变量已定义,则忽略此句
+= #附加,时即时变量还是延时变量取决于前面的定义
##Makefile函数
$(foreach var,list,text) for循环,对list中的没一个var执行text操作;
$(filter patten…,text) 在text中取出符合pattern格式的值
$(filter-out patten…,text) 在text中取出不符合pattern格式的值
$(wildcard pattern) 对于格式pattern中,wildcard取出真实存在的文件
$(patsubst pattern,replacement, $(var)) 循环var每一个元素,将符合pattern格式的用replacement替换
$(notdir $(SFILES)) 把SFILES中的目录部分去除,只保留文件名部分
如果SFILES是src/file1.c src/dir/file2.c include/header.h
执行notdir后就会是 file1.c file2.c header.h
##关于gcc生成依赖
gcc -M b.c
会把b.c头文件的查找路径都罗列出来,而这些路径应该是makefile的命令中需要添加的.
gcc -M -MF b.d b.c
把b.c的依赖文件写入b.d中
在命令中就使用下面语句:
gcc -c -o c.o c.c -MD -MF c.d
在编译中既能生成c.d,又能根据c.d中的依赖文件和目标依赖来生成c.o
CFLAGS 看可以取代加载参数,我觉得就是用了这个变量替换参数部分(
可以是CFLAGS = -Werror -I.
-Werror把警告都当作错误报告
-Ixxx把xxx当作编译该命令的头文件路径,就是人为指定头文件目录啦