- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我一直在构建一个 C++11
库,头文件/源文件的数量已经增长到编译程序调用它的程度,需要传递 20+ .cpp
文件到 g++
。我一直在阅读共享库,它似乎是最好的解决方案。
但是,由于 header /源代码经常更改,我希望创建一个 makefile
,它会自动从 header 和源代码生成所有 .so
文件。
为了更好地展示我正在尝试做的事情,我将采用我的子库之一 Chrono
并展示我将如何手动执行此操作。
我首先像这样创建目标文件,
$ g++ -std=c++11 -fPIC -g -c -Wall ../src/Chrono/cpp/DateTime.cpp
$ g++ -std=c++11 -fPIC -g -c -Wall ../src/Chrono/cpp/Schedule.cpp
$ g++ -std=c++11 -fPIC -g -c -Wall ../src/Chrono/cpp/Duration.cpp
$ g++ -std=c++11 -fPIC -g -c -Wall ../src/Chrono/cpp/DayCount.cpp
所以我现在有 DateTime.o
、Schedule.o
、Duration.o
和 DayCount.o
在当前目录中。然后我创建了 .so
文件,
$ g++ -shared -Wl,-soname,libChrono.so.1 -o libChrono.so.1.0.1 DateTime.o Schedule.o Duration.o DayCount.o -lc
然后我去,
$ rm ./*.o && ldconfig -n ./
因此我的工作目录现在包含 libChrono.so.1.0.1
和符号链接(symbolic link) libChrono.so.1
。
我需要为相当多的子目录执行此操作,因此您可以看到,只要对 header /源进行更改,这就会迅速变得低效。如果有人能帮我设计一个 makefile,只需调用 make
即可完成所有这一切,我将不胜感激。
谢谢!
更新:
根据 goldilock 的建议和一些挖掘,我设法组合在一起:
CXX=g++
CFLAGS=-std=c++11
TARGET=./lib/libChrono.so.1.0.1
CHRONODIR=./src/Chrono
CHRONOSRC=$(wildcard $(CHRONODIR)/cpp/*.cpp)
CHRONOOBJ=$(join $(addsuffix ../obj/, $(dir $(CHRONOSRC))), $(notdir (CHRONOSRC:.cpp=.o)))
all: $(TARGET)
@true
clean:
@-rm -f $(TARGET) $(CHRONOOBJ)
./lib/libChrono.so.1.0.1: $(CHRONOOBJ)
@echo "======================="
@echo "Creating library file $@"
@echo "======================="
@$(CXX) -shared -Wl,-soname,$(join $(basename $@), .1) -o $@ $^ -l
@echo "-- $@ file created --"
$(CHRONODIR)/cpp/../obj/%.o : $(CHRONOSRC)
@mkdir -p $(dir $@)
@echo "============="
@echo "Compiling $<"
@$(CXX) $(CFLAGS) -fPIC -g -Wall -c $< -o $@
4 .o
文件是在 lib/
中生成的,但我从 ld
收到了多个定义投诉。在我单独编译目标文件之前,但这会在一行中展开 CHRONOOBJ
。有什么想法吗?
最佳答案
幸运的是,您包含了问题的根源:
I've been building a C++11 library, and the number of header/source files has grown to the point where compiling programs invoking it, entails passing 20+ .cpp files to g++.
因为这揭示了一个潜在的XY problem .对此的直接解决方案是将目标文件放入存档(也称为 static 库)并使用它。
GNU make 有一个 implicit rule用于创建 C++ .o
文件。相当于:
%.o: %.cpp
$(CXX) $(CPPFLAGS) $(CXXFLAGS) -c $<
意思是,如果您在一个目录中制作 DateTime.o
,并且生成文件没有重新定义它,它将生成 DateTime.o
。但是,您可能想向 $(CXXFLAGS)
添加内容,例如:
CXXFLAGS += -Wall -Wextra --std=c++11
如果你打算坚持使用共享库路径,-fPIC
也可以去那里。这一行可能是您的整个 makefile。
但是,你还想把它们放在一起,所以你必须先声明所有的对象和组合它们的规则:
OBJS = DateTime.o Schedule.o Duration.o
libChrono.a: $(OBJS)
ar crvs $@ $^
最后一行(参见 man ar
)创建包含 $(OBJS)
中所有对象的存档 (libChrono.a)。然后,您可以通过将它放在同一目录(或库路径中的目录)并链接 -lChrono
来将它用于任何程序。只会提取和编译必要的部分。这样您就不必在系统目录中维护共享库。
如果您仍然认为需要共享库,$@
和$^
是automatic variables ;您可以使用类似的方法来创建一个 .so
,类似于:
SO_FLAGS = -shared
libChrono.so.1.0.1: $(OBJS)
$(CXX) $(SO_FLAGS) -Wl,-soname,libChrono.so.1 -o $@ $^ -lc
如果这是您的第一条规则,make
将处理所有事情:首先构建对象,然后构建库。请注意,这个已经排除了您的正常 $(CXXFLAGS)
以完全复制问题中的编译器行。
关于c++ - Makefile 构建共享库,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24829006/
我有一个 Makefile,它针对特定目标调用另一个 Makefile。假设主 Makefile 包含 some_dir/some_target: cd some_dir && make so
这两个文件大多出现在开源项目中。 它们的用途是什么?它们如何工作? 最佳答案 Makefile.am 是程序员定义的文件,由 automake 使用来生成 Makefile.in 文件( .am 代表
我的源代码位于一堆子目录中,例如: src/widgets/apple.cpp src/widgets/knob.cpp src/tests/blend.cpp src/ui/flash.cpp 在项
这就是我所拥有的: SUBDIRS = src/lib/ResourceManager all: $(SUBDIRS) $(SUBDIRS): make install -C $@ 我正在尝试
我想写一个 Makefile 来执行来自两个不同数组的两个输入的命令例如 a = A B C b = 1 2 3 ./run A 1 ./run B 2 ./run C 3 我不知道怎么写,因为在Ma
在 GNU make 手册的早期部分之一,Section 3.7 , 有一个 makefile 配方的大纲 immediate : immediate ; deferred defer
是否存在将 gmake 的 GNU Makefile 转换为可用于 make (FreeBSD-make) 的 Makefile 的实用程序? 最佳答案 该实用程序称为开发人员(程序员,制作大师,..
所以我前段时间了解了什么是 Makefile,创建了一个模板 Makefile,我所做的就是为我正在执行的每个程序复制和更改相同的文件。我改了几次,但它仍然是一个非常粗糙的Makefile。我应该如何
我正在做一些 Makefile 重构,并试图找出最简洁的方法来实现一个 Makefile,它执行以下操作: 有一个变量列出了所有源文件(可以是 C 和 C++ 文件) 所有目标文件都在 OBJ_DIR
我正在尝试创建一个 Makefile,它将通过 tic 编译位于目录中的 terminfo 文件。 tic 还将它自动创建的 termcap 文件复制到系统或用户特定的目标文件夹。对于普通用户,如果
我想要类似的东西 BROKEN_THINGS = \ thing1 \ # thing1 is completely broken thing2 \ # thing2 is broken to
如果我的程序必须为不同的结果(主要是错误)返回不同的值(例如 0、1、2、3 等),则调用该程序的 makefile 将不得不停止执行其余的 makefile 命令。即使该命令产生错误(返回非零值),
我正在学习使用漂亮的 Linux 工具:make。还有一点我想了解的: 让我们看一下这个简单的例子: JADE = $(shell find pages/*.jade) HTML = $(JADE:.
假设您有一个包含两个伪目标“all”和“debug”的 Makefile。 'debug' 目标旨在构建与 'all' 相同的项目,除了一些不同的编译开关(例如 -ggdb)。由于目标使用不同的编译开
我有一个调用多个其他生成文件的生成文件。 我想将 -j 参数传递给其他 makefile 调用。 类似(make -j8): all: make -f libpng_linux.mk -j
我处理过的 Makefile 大部分都很复杂,并且隐藏了很多关系。我自己从来没有写过一个,想知道是否有人有一些关于编写易于阅读和可重用的 Makefile 的提示? 最佳答案 我通常使用这样的东西,在
嘿,我有一个简单的“主” Makefile,它只是调用其他 makefile。我正在尝试执行以下操作,以便以正确的顺序构建组件: LIB_A = folder_a LIB_B = folder_b L
生成文件: #there is a whitespace after "/my/path/to" FOO = "/my/path/to" BAR = "dir" INCLUDE_DIRS = $(FO
我正在学习 makefile,我知道如何创建一个简单的 makefile。我正在继续使用嵌套的 makefile。这是我的目录结构 /src ...makefile ...main.cpp ...fo
什么TEMP0_FILES下面计算到? SOURCE_FILES可以等于多个源文件。请告诉我以下语法 :.cpp=.o 的用途 SOURCE_FILES = main.cpp TEMP0_FILES
我是一名优秀的程序员,十分优秀!