Android系统开发(3)——Makefile的编写

Makefile是什么?

makefile的作用:

1、project文件组织,编译成复杂的程序

2、安装及卸载我们的程序

Makefile使用演示样例

在/home/username/makefile文件夹下有例如以下三个文件:

Android系统开发(3)——Makefile的编写

main.c

#include <stdio.h>
#include <stdlib.h> int main(int argc, char *argv[]){
int x, y;
sscanf(argv[1], "%u", &x);
sccanf(argv[2], "%u", &y);
printf("func1:%u\n", func1(x, y));
printf("func2:%u\n", func2(x, y)); return 0;
}

func1.c

#include <stdio.h>

int func1(int x, int y){
return x + y;
}

func2.c

#include <stdio.h>

int func2(int x, int y){
return x + y;
}

以下我们開始编译上面的文件:

Android系统开发(3)——Makefile的编写

写好的Makefile文件例如以下:

obj=main.c func1.c func2.c
hello2:$(obj)
gcc $^ -o $@
.PHONY:clean
clean:
rm hello2
install:
cp hello2 /usr/local

运行make

Android系统开发(3)——Makefile的编写

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvZGF3YW5nYW5iYW4=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" alt="" />

Makefile的编写规则

Android系统开发(3)——Makefile的编写

Makefile由若干条上面的规则构成。每一个规则例如以下:

taget目标:prequisites(依赖)

command(命令)

注意:第二行必须有一个Tab缩进

比如上面两行表示要生成目标文件hello,在生成hello的同一时候须要依赖main.o func1.o func2.o这三个文件。并且运行gcc main.o func1.o func2.o -o hello命令来生成。

#建立第一个Makefile文件
hello:main.c func1.c func2.c
gcc main.c func1.c func2.c -o hello

我们来改动一下上面的Makefile

#建立第二个Makefile文件
hello:main.o func1.o func2.o
gcc main.o func1.o func2.o -o hello
main.o:main.c
gcc -c main.c
func1.o:func1.c
gcc -c func1.c
func2.o:func2.c
gcc -c func2.c

Android系统开发(3)——Makefile的编写

发现不仅生成了hello并且多了几个func1.c func1.o func2.c func2.o main.c main.o文件,以下我们改动一下Makefile文件。加入一个伪目标

hello:main.o func1.o func2.o
gcc main.o func1.o func2.o -o hello
main.o:main.c
gcc -c main.c
func1.o:func1.c
gcc -c func1.c
func2.o:func2.c
gcc -c func2.c clean:
rm func1.o func2.o main.o

clean就是一个伪目标,不会生成新的文件,以下我们来运行一下make clean来看一下效果

Android系统开发(3)——Makefile的编写

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvZGF3YW5nYW5iYW4=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" alt="" />

运行make clean后func1.o func2.o main.o全部删除了(能够想到卸载程序的原理)。以下我们就来加入一个install和uninstall伪目标

Android系统开发(3)——Makefile的编写

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvZGF3YW5nYW5iYW4=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" alt="" />

在我们运行make install的时候会将我们用到的运行文件和库文件复制到指定文件夹,在运行uninstall的时候会删除安装时的复制文件。

以下我们来使用变量MObj取代main.o func1.o func2.o

MObj = main.o func1.o func2.o
hello:$(MObj)
gcc $(MObj) -o hello
main.o:main.c
gcc -c main.c
func1.o:func1.c
gcc -c func1.c
func2.o:func2.c
gcc -c func2.c clean:
rm $(MObj)
install:
cp hello /usr/local/hello
uninstall:
rm /usr/local/hello

假设这样写MObj := main.o func1.o func2.o 表示不递归变量。

也能够使用系统的提前定义变量,常见的提前定义变量例如以下:

AR_____库文件维护程序的名称,默认值为ar

AS_____汇编程序的名称,默认值为as

CC_____C编译器的名称。默认值为cc

CXX____C++编译器的名称,默认值为g++

ARFLAGS_____库文件维护程序选项,无默认值

ASFLAGS_____汇编程序选项,无默认值

CFLAGS______C编译器选项,无默认值

CXXFLAGS____C++编译器选项,无默认值

以下我们使用提前定义变量来改写一下我们上面的Makefile文件

Android系统开发(3)——Makefile的编写

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvZGF3YW5nYW5iYW4=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" alt="" />

这样做的优点就是在我们换编译器的时候很方便。

以下我们来看看makefile中的自己主动变量及环境变量:

$*_______不包括扩展名的目标文件名

$<_______第一个依赖文件名

$?_______全部时间戳比目标文件晚的依赖文件

$@______目标文件完整名称

$^_______全部不反复的依赖文件

这个时候我们应该清楚了刚開始写的那个Makefile文件的内容了

Android系统开发(3)——Makefile的编写

上面还对伪目标进行了声明(.PHONY)这样做的目的是为了避免和其它文件同名冲突。

上一篇:Android开发笔记之《Window下安装Ubuntu双系统,Grub无法显示Window选项》


下一篇:php 168任意代码执行漏洞之php的Complex (curly) syntax