Linux系统编程 38 -makefile 3个自动变量和模式规则
学习笔记
自动变量
$@
在规则命令中,表示目标。
要注意:$@只能出现在命令上,不能出现在依赖和目标上。
$<
1.在规则的命令中,表示第一个依赖条件 (1个)
2.如果将该变量应用在模式规则中,它可将依赖条件中的依赖依次取出,套用模式规则 (多个)
$^
在规则的命令中,表示所有的依赖关系
makefile改写为:
$cat makefile
src=$(wildcard ./*.c) #add.c div1.c sub.c hello.c
obj=$(patsubst %.c,%.o,$(src)) #add.o div1.o sub.o hello.o
All:a.out
a.out:$(obj)
gcc $^ -o $@
hello.o:hello.c
gcc -c $< -o $@
add.o:add.c
gcc -c $< -o $@
sub.o:sub.c
gcc -c $< -o $@
div1.o:div1.c
gcc -c $< -o $@
clean:
-rm -rf $(obj) a.out
make
$make clean
rm -rf ./add.o ./div1.o ./hello.o ./sub.o a.out
$make
gcc -c add.c -o add.o
gcc -c div1.c -o div1.o
gcc -c hello.c -o hello.o
gcc -c sub.c -o sub.o
gcc add.o div1.o hello.o sub.o -o a.out
可以再省略
模式规则
目标的定义需要有"%"字符。"%"的意 思是表示一个或多个任意字符。在依赖目标中同样可以使用"%",只是依赖目标中的"%"的取值,取决于其目标。
%.o:%.c
gcc -c $< -o $@
修改makefile
$cat makefile
src=$(wildcard ./*.c) #add.c div1.c sub.c hello.c
obj=$(patsubst %.c,%.o,$(src)) #add.o div1.o sub.o hello.o
All:a.out
%.o:%.c
gcc -c $< -o $@
a.out:$(obj)
gcc $^ -o $@
clean:
-rm -rf $(obj) a.out
$make clean
rm -rf ./add.o ./div1.o ./hello.o ./sub.o a.out
$make
gcc -c add.c -o add.o
gcc -c div1.c -o div1.o
gcc -c hello.c -o hello.o
gcc -c sub.c -o sub.o
gcc add.o div1.o hello.o sub.o -o a.out
$./a.out
10+5 = 15
10-5 = 5
10/5 = 1
目的:方便扩展
mul.c 程序
$cat mul.c
int mul(int a,int b)
{
return (a*b);
}
hello.c 程序
$cat hello.c
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<pthread.h>
int add(int , int);
int sub(int , int);
int div1(int ,int);
int mul(int, int);
int main(int argc, char *argv[])
{
int a =10, b=5;
printf("%d+%d = %d\n",a,b,add(a,b));
printf("%d-%d = %d\n",a,b,sub(a,b));
printf("%d/%d = %d\n",a,b,div1(a,b));
printf("%d*%d = %d\n",a,b,mul(a,b));
return 0;
}
不需要修改makefile
$make clean
rm -rf ./add.o ./div1.o ./hello.o ./mul.o ./sub.o a.out
$make
gcc -c add.c -o add.o
gcc -c div1.c -o div1.o
gcc -c hello.c -o hello.o
gcc -c mul.c -o mul.o
gcc -c sub.c -o sub.o
gcc add.o div1.o hello.o mul.o sub.o -o a.out
$./a.out
10+5 = 15
10-5 = 5
10/5 = 1
10*5 = 50
静态模式规则:
$(obj):%.o:%.c
gcc -c $< -o $@
目的是:obj 找模式规则的时候,指定要找的是谁
$cat makefile
src=$(wildcard ./*.c) #add.c div1.c sub.c hello.c
obj=$(patsubst %.c,%.o,$(src)) #add.o div1.o sub.o hello.o
All:a.out
%.o:%.
gcc -S $< -o $@
%.o:%.c
gcc -c $< -o $@
a.out:$(obj)
gcc $^ -o $@
clean:
-rm -rf $(obj) a.out
这个时候,.o有两个来源,
有歧义,所以需要静态模式规则
创建新的文件clean
$touch clean
$make clean
make: `clean' is up to date.
现在出现问题了。
为什么?
当前目录下 有clean 或者All 文件的话
这个文件会影响make指令的判断
解决方法:
要把clean生成一个伪目标
在makefile中
添加:
.PHONY: clean All
$cat makefile
src=$(wildcard ./*.c) #add.c div1.c sub.c hello.c
obj=$(patsubst %.c,%.o,$(src)) #add.o div1.o sub.o hello.o
All:a.out
%.o:%.c
gcc -c $< -o $@
a.out:$(obj)
gcc $^ -o $@
clean:
-rm -rf $(obj) a.out
.PHONY: clean All
PS:phony:伪造的意思
$make clean
rm -rf ./add.o ./div1.o ./hello.o ./mul.o ./sub.o a.out
$make
gcc -c add.c -o add.o
gcc -c div1.c -o div1.o
gcc -c hello.c -o hello.o
gcc -c mul.c -o mul.o
gcc -c sub.c -o sub.o
gcc add.o div1.o hello.o mul.o sub.o -o a.out
为了方便编写,可以添加常用的参数
-Wall -g -I -L -l
//修改makefile
$cat makefile
src=$(wildcard ./*.c) #add.c div1.c sub.c hello.c
obj=$(patsubst %.c,%.o,$(src)) #add.o div1.o sub.o hello.o
myArgs= -Wall -g
All:a.out
%.o:%.c
gcc -c $< -o $@ $(myArgs)
a.out:$(obj)
gcc $^ -o $@ $(myArgs)
clean:
-rm -rf $(obj) a.out
.PHONY: clean All
$make clean
rm -rf ./add.o ./div1.o ./hello.o ./mul.o ./sub.o a.out
$make
gcc -c add.c -o add.o -Wall -g
gcc -c div1.c -o div1.o -Wall -g
gcc -c hello.c -o hello.o -Wall -g
gcc -c mul.c -o mul.o -Wall -g
gcc -c sub.c -o sub.o -Wall -g
gcc add.o div1.o hello.o mul.o sub.o -o a.out -Wall -g
$gdb a.out
GNU gdb (Ubuntu 7.7.1-0ubuntu5~14.04.2) 7.7.1
Copyright (C) 2014 Free Software Foundation, Inc.
License GPLv3+: GNU GPL version 3 or later <http://gnu.org/licenses/gpl.html>
This is free software: you are free to change and redistribute it.
There is NO WARRANTY, to the extent permitted by law. Type "show copying"
and "show warranty" for details.
This GDB was configured as "x86_64-linux-gnu".
Type "show configuration" for configuration details.
For bug reporting instructions, please see:
<http://www.gnu.org/software/gdb/bugs/>.
Find the GDB manual and other documentation resources online at:
<http://www.gnu.org/software/gdb/documentation/>.
For help, type "help".
Type "apropos word" to search for commands related to "word"...
Reading symbols from a.out...done.
(gdb) list 1
1 #include<stdio.h>
2 #include<stdlib.h>
3 #include<string.h>
4 #include<unistd.h>
5 #include<pthread.h>
6
7 int add(int , int);
8 int sub(int , int);
9 int div1(int ,int);
10 int mul(int, int);
(gdb)