makefile文件 1.实例:makefile.bak .PHONY:clean #显式指定clean 为伪目标,防止在当前目录下存在clean文件是无法执行清理工作 ping:ping.o main.o g++ -Wall -g main.o ping.o -o ping -lpthread ping.o:ping.cpp ping.h g++ -Wall -g -c ping.cpp -o ping.o main.o:main.cpp g++ -Wall -g -c main.cpp -o main.o clean: echo begin delete ... rm -f *.o 2.makefile自动化变量makefile .PHONY:clean #显式指定clean 为伪目标,防止在当前目录下存在clean文件是无法执行清理工作 OBJECTS=ping.o main.o ping:$(OBJECTS) #自定义变量 g++ -Wall -g $^ -o $@ -lpthread #$^ 依赖列表的所有项ping.o main.o $@ 表示目标项ping ping.o:ping.cpp ping.h g++ -Wall -g -c $< -o $@ #$<依赖列表的第一项ping.cpp main.o:main.cpp @g++ -Wall -g -c $< -o $@ #在命令前加@,表示不显示命令 clean: rm -f $(OBJECTS) #自动化变量和自定义变量的使用 makefile的执行:make clean -f makefile.bak -f 指定要执行的makefile文件 3.makefile默认推导规则 实例: 生成多个可执行文件 01Test.cpp 02Test.cpp .PHONY:clean all BIN=01Test 02Test all:$(BIN) clean: rm -f $(BIN) 执行后生成01Test 02Test可执行文件,系统执行的事隐含推导规则,也可以自己编写推导规则 模式规则:%.o:%.c 01Test.cpp 02Test.cpp .PHONY:clean all BIN=01Test 02Test all:$(BIN) %.o:%.c g++ -Wall -g -c $< -o $@ 01Test:01Test.o g++ -Wall -g $^ -o $@ 02Test:02Test.o g++ -Wall -g $^ -o $@ clean: rm -f *.o $(BIN) 后缀规则:.c.o: .PHONY:clean all BIN=01Test 02Test all:$(BIN) .c.o: g++ -Wall -g -c $< -o $@ 01Test:01Test.o g++ -Wall -g $^ -o $@ 02Test:02Test.o g++ -Wall -g $^ -o $@ clean: rm -f *.o $(BIN) 定义变量进一步简化makefile: .PHONY:clean all CC=g++ CFLAGS=-Wall -g BIN=01Test 02Test all:$(BIN) .c.o: $(CC) $(CFLAGS) -c $< -o $@ 01Test:01Test.o $(CC) $(CFLAGS) $^ -o $@ 02Test:02Test.o $(CC) $(CFLAGS) $^ -o $@ clean: rm -f *.o $(BIN)
多级目录makefile的实现:
1.make常用内嵌函数
2.二级目录的实现
实例
.PHONY:clean
#显式指定clean 为伪目标,防止在当前目录下存在clean文件是无法执行清理工作
CC =g++ CFLAGS =-Wall -g
BIN = ping SUBDIR =$(shell ls -d */)
ROOTSRC =$(wildcard *.cpp) #当前目录下匹配模式的文件
ROOTOBJ =$(ROOTSRC:%.cpp=%.o)
SUBSRC =$(shell find $(SUBDIR) -name '*.cpp') #指定目录下的源文件
SUBOBJ =$(SUBSRC:%.cpp=%.o) #模式替换函数
$(BIN):$(ROOTOBJ) $(SUBOBJ)
$(CC) $(CFLAGS) -o $(BIN) $(ROOTOBJ) $(SUBOBJ) -lpthread
.c.o:
$(CC) $(CFLAGS) -c $< -o $@
clean:
@rm -f $(BIN) $(ROOTOBJ) $(SUBOBJ)
#二级目录makefile文件的使用
3.多级目录、多个makefile的实现
SUBDIRS = Test1 Test2
.PHONY:default all clean $(SUBDIRS)
default:all
all clean:
$(MAKE) $(SUBDIRS) TARGET=$@
$(SUBDIRS):
$(MAKE) -C $@ $(TARGET)
#-C表示进入到目录 make -C Test1 all等价于make all Test1/makefile
Test1目录makefile
CC = g++
BIN = Test1
OBJS = Test1.o
.PHONY: all clean print
all:print $(BIN)
print:
@echo "-----------make all in $(PWD) -----------"
$(BIN):$(OBJS)
$(CC) $(OBJS) -o $@
%.o:%.cpp
$(CC) -c $<
clean:
@echo "-----------make clean in $(PWD) -----------"
rm -f $(BIN) $(OBJS)
Test2目录的makefile
CC = g++
BIN = Test2
OBJS = Test2.o
CFLAGS = -Wall -g
.PHONY: all clean print
all:print $(BIN)
print:
@echo "-----------make all in $(PWD) -----------"
$(BIN):$(OBJS)
$(CC) $(CFLAGS) $(OBJS) -o $@
%.o:%.cpp
$(CC) $(CFLAGS) -c $<
clean:
@echo "-----------make clean in $(PWD) -----------"
rm -f $(BIN) $(OBJS)
Linux 库的使用: 1.静态库(*.a)实质就是把.o文件的集合,归档打包 1.1生成.o文件:g++ -Wall hello_fn.c -o hello_fn.o 1.2生成.a文件:ar rcs libhello.a hello_fn.o (ar 是gnu归档工具,rcs('r' 静态库存在则替换replace,'c' 不存在则创建,'s' 保存.o文件的索引信息到库文件中去) 1.3 库文件的使用:g++ -Wall main.c libhello.a -o main 或者 g++ -Wall -L. main.c -o main -lhello 由于gcc编译器默认不会搜索当前目录,索引用 -L. 在当前目录搜索库文件。 2.动态库(*.so or *.sa) 2.1生成.o文件:g++ -c Test.cpp 2.2生成.so文件:g++ -shared -fPIC Test.o -o libTest.so 2.3使用动态库:g++ -L. main.o - o main lTest. 当动态库和静态库同时存在是,优先使用动态库。 2.4程序运行动态库加载路径。 a. 拷贝.so文件到系统共享库/usr/lib/路径下。 b. 修改系统环境变量,增加库的路径。vi~/bash_profile LD_LIBRARY_PATH=库的路径(当前用户配置)。 c. ldconfig 配置ld.so.conf,ldconfig更新ld.so.cache vi /etc/ld.so.conf配置库文件路径(全局配置) 2.5 lld查看文件包含了哪些库文件。 3.库搜索路径: 3.1 C_INCLUDE_PATH、LIBRARY_PATH 3.2 -I 指定头文件目录、-L指定库文件搜索目录 3.3 环境变量指定的目录。 vi ~/.bash_profile export C_INCLUDE_PATH = /opt/soft/include/ export LIBRARY_PATH=/opt/soft/lib/ .~/.bash_profile 执行一下生效。 ~/ 当前用户根目录 /系统根目录 3.4系统指定的目录。一般系统默认的库头文件和库文件默认在 /usr/include/ /usr/lib/