声明:本文写于 狄泰软件学院makefile课程 学后总结,本文版权归狄泰软件学院所有!
疯狂想法的注意事项:
当 .dep 文件生成后, 如果动态改变头文件的依赖关系, 那么 make 可能无法检测到这个改变, 进而做出编译决策。
解决方案:
将依赖文件名作为目标加入自动生成依赖关系中
通过 include 加载依赖文件时判断是否执行规则
在规则执行时重新生成依赖关系文件
最后加载新的依赖文件
.PHONY : all clean
MKDIR := mkdir
RM := rm -fr
CC := gcc
DIR_DEPS := deps
DIR_EXES := exes
DIR_OBJS := objs
DIRS := $(DIR_DEPS) $(DIR_EXES) $(DIR_OBJS)
EXE := app.out
EXE := $(addprefix $(DIR_EXES)/, $(EXE))
SRCS := $(wildcard *.c)
DEPS := $(SRCS:.c=.dep)
DEPS := $(addprefix $(DIR_DEPS)/, $(DEPS))
OBJS := $(SRCS:.c=.o)
OBJS := $(addprefix $(DIR_OBJS)/, $(OBJS))
all : $(DIR_OBJS) $(DIR_EXES) $(EXE)
ifeq ("$(MAKECMDGOALS)", "all")
-include $(DEPS)
endif
ifeq ("$(MAKECMDGOALS)", "")
-include $(DEPS)
endif
$(EXE) : $(OBJS)
$(CC) -o $@ $^
@echo "Success! Target => $@"
$(DIR_OBJS)/%.o : %.c
$(CC) -o $@ -c $(filter %.c, $^)
$(DIRS) :
$(MKDIR) $@
ifeq ("$(wildcard $(DIR_DEPS))", "")
$(DIR_DEPS)/%.dep : $(DIR_DEPS) %.c
else
$(DIR_DEPS)/%.dep : %.c
endif
@echo "Creating $@ ... "
@set -e; \
$(CC) -MM -E $(filter %.c, $^) | sed 's,\(.*\)\.o[ :]*,objs/\1.o $@ : ,g' > $@
clean :
$(RM) $(DIRS)
rebuild :
@$(MAKE) clean
@$(MAKE) all
小结:
- makefile 中可以将目标的拆分写到不同的地方
- include 关键字能够触发相应规则的执行
- 如果规则的执行导致依赖更新, 可能导致再次解释执行相应规则
- 依赖文件需要依赖源文件得到正确的编译决策
- 自动生成文件间的依赖关系能够提高makefile的移植性