不必要地重新编译目标

时间:2012-01-27 13:54:59

标签: compilation makefile

首先,这是我的Makefile,它真的很酷。 它为当前目录中的每个“c”文件生成可执行文件。 琐事与贝壳,PITA与Make ......我提到过,我已经十年没有了所以请不要笑......

CC = gcc
#List all 'c' files by wildcard.
SOURCES=$(wildcard *.c)
#Get the 'exe' equivalent of said file
EXECUTABLES=$(SOURCES:%.c=%)

#For each of the '*.exe' files
#append a 'c' suffix to the target
build: $(EXECUTABLES)
    $(CC) -c $(<:%=%.c) -o $<

clean:
    find ./ ! -name "*.c" -a ! -name "Makefile" -type f -delete

rebuild: clean build

以下是我的目录的内容:

% ls -la
total 32
drwxr-xr-x  2 yomom yomom 4096 2012-01-27 13:38 ./
drwx------ 94 yomom yomom 4096 2012-01-27 13:38 ../
-rw-r--r--  1 yomom yomom  990 2012-01-27 12:47 array_example.c
-rw-r--r--  1 yomom yomom  341 2012-01-27 13:38 Makefile
-rw-r--r--  1 yomom yomom  430 2011-12-05 13:08 pointers.c
-rw-r--r--  1 yomom yomom  319 2012-01-27 12:45 should_create_warning.c
-rw-r--r--  1 yomom yomom 1472 2011-12-19 16:16 socket-client.c
-rw-r--r--  1 yomom yomom 1150 2011-12-19 16:15 socket-server.c

现在我运行make

% make
gcc     array_example.c   -o array_example
gcc     pointers.c   -o pointers
gcc     should_create_warning.c   -o should_create_warning
gcc     socket-client.c   -o socket-client
gcc     socket-server.c   -o socket-server
gcc -c array_example.c -o array_example

很好,它重新编译了所有内容,但最后一行输出看起来有些不一致。 怎么会和其他人不一样呢?

% ls -la
total 68
drwxr-xr-x  2 yomom yomom 4096 2012-01-27 13:38 ./
drwx------ 94 yomom yomom 4096 2012-01-27 13:38 ../
-rw-r--r--  1 yomom yomom 1428 2012-01-27 13:38 array_example
-rw-r--r--  1 yomom yomom  990 2012-01-27 12:47 array_example.c
-rw-r--r--  1 yomom yomom  341 2012-01-27 13:38 Makefile
-rwxr-xr-x  1 yomom yomom 7164 2012-01-27 13:38 pointers*
-rw-r--r--  1 yomom yomom  430 2011-12-05 13:08 pointers.c
-rwxr-xr-x  1 yomom yomom 7139 2012-01-27 13:38 should_create_warning*
-rw-r--r--  1 yomom yomom  319 2012-01-27 12:45 should_create_warning.c
-rwxr-xr-x  1 yomom yomom 7515 2012-01-27 13:38 socket-client*
-rw-r--r--  1 yomom yomom 1472 2011-12-19 16:16 socket-client.c
-rwxr-xr-x  1 yomom yomom 7579 2012-01-27 13:38 socket-server*
-rw-r--r--  1 yomom yomom 1150 2011-12-19 16:15 socket-server.c

我再次运行make,期待它到NOP,毕竟,我没有改变任何东西

% make
gcc -c array_example.c -o array_example

但是它产生了输出,是什么给出了?

% ls -la
total 68
drwxr-xr-x  2 yomom yomom 4096 2012-01-27 13:51 ./
drwx------ 94 yomom yomom 4096 2012-01-27 13:51 ../
-rw-r--r--  1 yomom yomom 1428 2012-01-27 13:51 array_example
-rw-r--r--  1 yomom yomom  990 2012-01-27 12:47 array_example.c
-rw-r--r--  1 yomom yomom  341 2012-01-27 13:38 Makefile
-rwxr-xr-x  1 yomom yomom 7164 2012-01-27 13:38 pointers*
-rw-r--r--  1 yomom yomom  430 2011-12-05 13:08 pointers.c
-rwxr-xr-x  1 yomom yomom 7139 2012-01-27 13:38 should_create_warning*
-rw-r--r--  1 yomom yomom  319 2012-01-27 12:45 should_create_warning.c
-rwxr-xr-x  1 yomom yomom 7515 2012-01-27 13:38 socket-client*
-rw-r--r--  1 yomom yomom 1472 2011-12-19 16:16 socket-client.c
-rwxr-xr-x  1 yomom yomom 7579 2012-01-27 13:38 socket-server*
-rw-r--r--  1 yomom yomom 1150 2011-12-19 16:15 socket-server.c

它重新编译了array_example文件,为什么总是那个?

解决方案(感谢Eldar Abusalimov)

CC = gcc
#List all 'c' files by wildcard.
SOURCES=$(wildcard *.c)
#Get the 'exe' equivalent of said file
EXECUTABLES=$(SOURCES:%.c=%)

all: $(EXECUTABLES)
.PHONY: all
$(EXECUTABLES): % : %.c
    $(CC) -c $< -o $@

.PHONY clean:
clean:
    find ./ ! -name "*.c" -a ! -name "Makefile" -type f -delete

rebuild: clean all

最后的会议

删除所有生成的文件

% make clean 
find ./ ! -name "*.c" -a ! -name "Makefile" -type f -delete

% ls -la
total 48
drwxr-xr-x  2 yomom yomom 4096 2012-01-27 18:07 ./
drwx------ 94 yomom yomom 4096 2012-01-27 18:07 ../
-rw-r--r--  1 yomom yomom  990 2012-01-27 17:38 array_example.c
-rw-r--r--  1 yomom yomom  428 2012-01-27 17:38 array_of_pointers.c
-rw-r--r--  1 yomom yomom  274 2012-01-27 17:38 const_ptr.c
-rw-r--r--  1 yomom yomom  293 2012-01-27 17:38 function_pointers.c
-rw-r--r--  1 yomom yomom  313 2012-01-27 18:06 Makefile
-rw-r--r--  1 yomom yomom  430 2012-01-27 17:38 pointers.c
-rw-r--r--  1 yomom yomom  228 2012-01-27 17:38 pointer_to_constant.c
-rw-r--r--  1 yomom yomom  253 2012-01-27 17:38 pointer_to_pointer.c
-rw-r--r--  1 yomom yomom 1472 2012-01-27 17:38 socket-client.c
-rw-r--r--  1 yomom yomom 1150 2012-01-27 17:38 socket-server.c

再次运行make

% make
gcc -c array_example.c -o array_example
gcc -c array_of_pointers.c -o array_of_pointers
gcc -c const_ptr.c -o const_ptr
gcc -c function_pointers.c -o function_pointers
gcc -c pointers.c -o pointers
gcc -c pointer_to_constant.c -o pointer_to_constant
gcc -c pointer_to_pointer.c -o pointer_to_pointer
gcc -c socket-client.c -o socket-client
gcc -c socket-server.c -o socket-server

现在它不再做愚蠢的事了......

% make
make: Nothing to be done for `all'.

1 个答案:

答案 0 :(得分:3)

只需删除build: $(EXECUTABLES)的食谱即可。首先,这样的规则不应该创建build文件。其次,回答你的问题,为什么只有array_example总是被重新编译,$(<:%=%.c)会产生第一个列出的先决条件,这可能是wildcard返回的第一个先决条件(哪个排序)字典顺序的结果,不保证,但usually it does)。也就是说,

SOURCES=$(wildcard *.c)# returns 'array_example.c pointers.c ...'.
EXECUTABLES=$(SOURCES:%.c=%)# is 'array_example pointers ...'.

build: $(EXECUTABLES)
# Here '$<' is the first item of '$(EXECUTABLES)', which is 'array_example'.
# And '$(<:%=%.c)' returns 'array_example.c'
    $(CC) -c $(<:%=%.c) -o $<

最后,不要忘记添加.PHONY目标。

.PHONY: build
build: $(EXECUTABLES)

使用an implicit rule构建$(EXECUTABLES)中列出的所有程序,以便从C源生成可执行文件。无需指定任何其他内容。

如果隐式规则不正常,您可以覆盖它,例如如下:

$(EXECUTABLES) : % : %.c
    $(CC) -o $@ $<