从gcc到Makefile简易版

1、Makefile的应用
  我们主要用它来编译源代码,生成结果代码,然后把结果代码连接起来生成可执行文件或者库文件。
2、Makefle简单例子的深入学习
  程序概述:为了连接makefile的流程,我将最简单的程序“hello wrold!”分成两个.c文件,并最终在main中进行编译。
  以下都是对同个文件进行makefile的编译,知识makfile编写回比较贴近shell脚本的语法
  如果没有指定输出项目的时候Make会自动找到makefile中第一个目标中没有通配符的目标进行构造,所以步骤是:
  例子1:
  终端界面的命令格式

  gcc -o hello main.c hello.c wrold.c

  注释:gcc 代表的是编译器类型

  -o 表示生成可执行文件
  hello 表示生成的可执行文件的名称
  *.c 表示c程序文件


  例子2:
  接下来我们编写Makefile文件

  hello: main.c hello.c wrold.c
  gcc -o hello main.c hello.c wrold.c

  注释:它主要分成了三个部分

  第一行冒号之前的hello,我们称之为目标(target),被认为是这条语句所要处理的对象,具体到这里就是我们所要编译的这个程序hello。
  冒号后面的部分(main.c hello.c wrold.c),我们称之为依赖关系表,也就是编译hello所需要的文件,(此部分是可以省略的)
  命令部分,相信你也看得出这就是一条编译命令。现在我们只要将上面这两行语句写入一个名为Makefile或者makefile的文件,然后在终端中输入make命令,就会看到它按照我们的设定去编译程序了。
  请注意,在第二行的“gcc”命令之前必须要有一个tab缩进。语法规定Makefile中的任何命令之前都必须要有一个tab缩进,否则make就会报错。
  

  例子3:
  接下来我们使用shell脚本语言对于makefile进行更改

  cc = gcc
  prom = hello
  source = main.c hello.c wrold.c

  $(prom): $(source)
  $(cc) -o $(prom) $(source)

  注释:$(名字) 代表一个变量,shell脚本在执行这个变量时,会进行替换

  cc、prom以及source。它们分别告诉了make我们要使用的编译器、要编译的目标以及源文件
  

  例子4:
  对于一个文件的修改,需要重新编译整个流程(其他的.c文件也需要编译)

  cc = gcc
  prom = hello
  objs = main.o hello.o wrold.o

  $(prom): $(objs)
  $(cc) -o $(prom) $(objs)

  main.o: main.c
  $(cc) -c main.c

  hello.o: hello.c
  $(cc) -c hello.c

  wrold.o: wrold.c
  $(cc) -c wrold.c

  注释:标准的编译过程中,源文件往往是先被编译成目标文件,然后再由目标文件连接成可执行文件的。我们可以利用这一点来调整一下这些文件之间的依赖关系。

  总工程通过.o文件进行编译的话,如果.c文件没改变,那么原先的.o可以继续使用,不需要重新编译。
  

  例子5:
  我们发现许多类型重复的语句,这样的效率并不高,可以将类型一样的语句整合到一起

  cc = gcc
  prom = hello
  objs = main.o hello.o wrold.o

  $(prom): $(objs)
  $(cc) -o $(prom) $(objs)

  %.o:%.c
  $(cc) -c $< -o $@

  注释:makeefile中使用的”%“通配符和系统中“*”通配符的区别:

  %.o:%.c的的意思是,“%”代表通配符的意思,等价于% =任意字符(比如main,hello等),当然结尾的格式要一样
  $@:目标的名字($@代表的是当前语句的目标,即%.o)
  $^:构造所需文件列表所有所有文件的名字
  $<:构造所需文件列表的第一个文件的名字
  $?:构造所需文件列表中更新过的文件

  例子6:
  增加自动清理编译结果的功能就可以为其定义一个带伪目标的规则

  cc = gcc
  prom = hello
  objs = main.o hello.o wrold.o

  %.o:%.c
  $(cc) -c $< -o $@

  $(prom): $(objs)
  $(cc) -o $(prom) $(objs)

  #%.o:%.c 
  # $(cc) -c $< -o $@

  clean:
  rm -rf $(prom) $(objs)

 注释:有了上面最后两行代码,当我们在终端中执行make clean命令时,它就会去删除该工程生成的所有编译文件。

  而命令中的“$<”和“$@”则是自动化变量,“$<”表示所有的依赖目标集(也就是“foo.c bar.c”),“$@”表示目标集(也就是“foo.o bar.o”)
  

  例子7:
  自动扫描当前文件夹下的.c或.h文件

  cc = gcc
  prom = hello
  deps = $(shell find ./ -name "*.h")
  src = $(shell find ./ -name "*.c")
  objs = $(src:%.c=%.o)

  $(prom): $(objs)
  $(cc) -o $(prom) $(objs)

  %.o:%.c
  $(cc) -c $< -o $@

  clean:
  rm -rf $(prom) $(objs)

  注释:$(shell find ./ -name "*.h") //意思是寻找当前文件夹下的.h文件

    $(src:%.c=%.o) //将src中的.c结尾的变量名字替换成.o结尾的名字

posted on 2019-10-10 14:43  kukou  阅读(192)  评论(0编辑  收藏  举报

导航