舒适提示:阅读本文的同窗最好能了解makefile和python的编写规则。python
不懂的同窗可以先保存在收藏夹。以便往后查看。mysql
事实上以前我一直很是懒,我不想了解makefile规则。因为在linux下开发我一直使用Qt creator。(很是多时候正是一些“懒人”的创造力。解放了咱们的双手,显然现在咱们还需要用双手写makefile)。Qt creator是一个很是好的IDE。而且可以跨平台开发。但是相比VS,显然仍是不够优秀。linux
所以很是多开发人员都会选择在Windows下开发C/C++程序,而后部署在Linux下运行。固然我也不例外。因此近期花了几个晚上了解makefile的编写规则。sql
開始的时候。我參照网上一些makefile的样例,写了一个第一版的makefile。然而这个makefile在编译个人project的时候报错。shell
主要出错体现在:数组
%.o: %.cppapp
$(CXX) -fpic -c $(INCPATH) $< -o $@函数
固然可以写成学习
$(objdir)/%.o:$(srcdir)/%.cppui
$(CXX) -fpic -c $(INCPATH) $< -o $@
缘由在于:
一、.o文件与.cpp文件处于不一样的文件夹下。
二、不一样的.o文件或不一样的.cpp文件处于不一样的文件夹下。
这时我找到两种解决方法:
一、就是用VPATH这个特殊变量,但是我不可能将所有要包括的文件夹都一一手动包括进来,因而我放弃。
二、就是把所有的编译规则列举出来。
我终于选择另一种解决方法。
因为以前遇到这个困难时,我特地去了解Qt生成的makefile(事实上这个makefile是根据.proproject文件生成的)。而这个makefile正是将所有的编译规则都列举出来。
因而就有如下这个python脚本。事实上開始的时候我想用shell来作这一步工做的,但是我看到sed和awk。我头都晕,以前还一直抵触学习sed和awk。所以最后选择了python。
#encoding: utf-8 import os import os.path import sys #sys.exit(0); ######################################################################################################################### # # 本脚本的做用是:经过配置必要的信息,用python来生成makefile。(技术支持:www.guimigame.com) # @FILENAME 运行脚本输出makefile文件名称 # @BIN 生成可运行文件名称 # @SUFFIX 源文件后缀 # @ROOTPATH “根”文件夹路径(脚本工做文件夹的上一层) # @PWD 当前工做文件夹 # @WD 工做文件夹。假设程序有多个工做文件夹请一一用append加上 # @BINDIR 可运行文件输出文件夹 # @OBJDIR 中间文件输出文件夹 # @INCROOTPATH 头文件包括路径的“根路径”,方便INCPATH的编写 # @LIBROOTPATH 包括库的“根”路径。方便LIBS的编写 # @INCPATH 头文件包括路径 # @SYSLIBS 包括的系统库 # @LIBS 编译程序需要包括的库 # @CXX 通常填写gcc/g++ # @FLAGS gcc/g++的编译标志 # ######################################################################################################################### FILENAME = 'makefile'; BIN = "DatabaseServer"; SUFFIX = ".cpp"; ROOTPATH = os.path.dirname(os.path.dirname(os.path.abspath(__file__))); PWD = os.getcwd(); WD = []; WD.append(PWD); WD.append(ROOTPATH + "/common"); BINDIR = PWD + "/Bin/"; OBJDIR = BINDIR + "obj/"; INCROOTPATH = "-I " + ROOTPATH; LIBROOTPATH = "-L " + ROOTPATH; INCPATH = INCROOTPATH + "/common" + " " + INCROOTPATH + "/lib/include"; SYSLIBS = " -lmysqlclient -lpthread" LIBS = LIBROOTPATH + "/lib/linux " + "-lTimeManager -lServerConfig -lGameSocket -lCommon -lTinyxml" + SYSLIBS; CXX = "g++"; FLAGS = '-g -Wall'; ######################################################################################################################### # # 下面不需要再配置 # ######################################################################################################################### OBJFILE = ''; OBJ2SRC = []; SOURCES = ""; def SearchFiles(path): global OBJFILE; global OBJ2SRC; global SOURCES; global SUFFIX; listFile = os.listdir(path); for file in listFile: if os.path.isdir(os.path.join(path, file)): SearchFiles(os.path.join(path, file)); elif file.endswith(SUFFIX) > 0: OBJFILE = file; OBJFILE = OBJFILE.replace(SUFFIX,'.o'); OBJ2SRC.append([OBJDIR + OBJFILE,path + "/" + file]); SOURCES += path + "/" + file + " "; for dir in WD: SearchFiles(dir); if os.path.exists(FILENAME): os.remove(FILENAME); f = open(FILENAME,'w'); f.write("PWD = " + PWD + "\n"); f.write("CXX = " + CXX + "\n"); f.write("INCPATH = " + INCPATH + "\n"); f.write("LIBS = " + LIBS + "\n"); f.write("BINDIR = " + BINDIR +"\n"); f.write("OBJDIR = " + OBJDIR + "\n"); f.write("BIN = " + BIN + "\n"); f.write("SOURCES = " + SOURCES+ "\n"); f.write("SOURCEFILES = $(notdir $(SOURCES))\n"); f.write("OBJECTS = $(addprefix $(OBJDIR), $(patsubst %.cpp,%.o,$(SOURCEFILES)))\n"); f.write("FLAGS = " + FLAGS + "\n"); f.write("\n"); f.write("all:makedir $(OBJECTS)\n"); f.write(" $(CXX) $(FLAGS) $(INCPATH) -o $(BIN) $(OBJECTS) $(LIBS);\n"); f.write("\n"); f.write("makedir:\n"); f.write(' $(shell if [ -n "$(OBJDIR)" -a ! -e "$(OBJDIR)" ];then mkdir -p $(OBJDIR); fi)\n'); f.write(' $(shell if [ -n "$(BINDIR)" -a ! -e "$(BINDIR)" ];then mkdir -p $(BINDIR); fi)\n'); f.write("\n"); for val in OBJ2SRC: f.write(val[0] + ":" + val[1] +"\n"); f.write(" rm -f $@\n"); f.write(" $(CXX) -fpic -c $(INCPATH) $< -o $@\n"); f.write("\n"); f.close(); os.system("make"); os.system("mv " + BIN + " " + BINDIR); os.system("cd " + OBJDIR + ";rm -f *.o");
怎样编写makefile和python,这里不做说明。因为这篇文章不是makefile和python的教程。下面要说明的是SearchFiles函数。
经过遍历以前设定的project工做文件夹。调用SearchFiles遍历该文件夹下所有的源文件(.cpp),及设定目标文件(.o)的绝对路径,终因而tuple的形式保存到OBJ2SRC数组中。还有的是将所有源文件保存在SOURCES中。固然这个过程当中会递归遍历所有子文件夹,查找到所有的源文件。终于在for val in OBJ2SRC:遍历所有的数据;列出所有的源文件(.cpp)生成所相应的目标文件(.o)。将编译规则写进makefile。
这是我要编译的project。固然截图仅仅是其中一部分。
这个project需要包括的文件除了在DatabaseServer下,还要包括在../common其中(脚本中代码WD.append(ROOTPATH + "/common");)。
我截图是为了证实。这个脚本是可行的。有人可能会说为何不写一个測试样例。
事实上我想说。很是多时候要弄懂一些技术。动手去作或许是最好的方法。假设你有什么问题,欢迎与我讨论!