c++ - 使用 automake 将子目录编译到库中的最佳方法?

标签 c++ autotools automake libtool

我有以下文件

├── configure.ac
├── main.cpp
├── Makefile.am
└── procs
    ├── Makefile.am
    ├── proc1
    │   └── subprocs
    │       ├── subprocA
    │       │   ├── ProcA_SubprocA.cc
    │       │   └── ProcA_SubprocA.h
    │       └── subprocB
    │           ├── ProcA_SubprocB.cc
    │           └── ProcA_SubprocB.h
    ├── proc2
    │   └── subprocs
    │       ├── subprocA
    │       │   ├── ProcB_SubprocA.cc
    │       │   └── ProcB_SubprocA.h
    │       └── subprocB
    │           ├── ProcB_SubprocB.cc
    │           └── ProcB_SubprocB.h
    └── Subprocs.h

我的目标是从 procs 中的所有内容创建一个库,以便它可以在 ma​​in.cpp 中使用,它看起来像这样:

main.cpp

#include "Subprocs.h"

int main(){

    ProcA_SubprocA * p = new ProcA_SubprocA();
    p->use();
    delete p;

    return 0;
}

我想在 Subprocs.h 中包含子过程的所有内容,因此它可以用作某种接口(interface):

子进程.h

#include "ProcA_SubprocA.h"
#include "ProcA_SubprocB.h"
#include "ProcB_SubprocA.h"
#include "ProcB_SubprocB.h"

我的configure.ac 如下所示:

配置.ac

AC_PREREQ([2.69])
AC_INIT([main], [0.1])
AM_INIT_AUTOMAKE([foreign subdir-objects])

#AC_CONFIG_MACRO_DIR([m4]) 
#LT_INIT

AC_PROG_CXX
AC_PROG_CC
AC_PROG_RANLIB

AC_CONFIG_FILES([Makefile procs/Makefile])
AC_OUTPUT

和我的Makefile.am是这样的

生成文件.am

SUBDIRS = procs

AM_CPPFLAGS = -Iprocs

bin_PROGRAMS = main
main_SOURCES = main.cpp
#added:
main_LDADD = procs/libprocs.a

#ACLOCAL_AMFLAGS = -I m4

procs/Makefile.am

AM_CPPFLAGS = -Iproc1/subprocs/subprocA -Iproc1/subprocs/subprocB \
              -Iproc2/subprocs/subprocA -Iproc2/subprocs/subprocB

#lib_LIBRARIES = libprocs.la
noinst_LIBRARIES = libprocs.a
libprocs_a_SOURCES = Subprocs.h \
                      proc1/subprocs/subprocA/ProcA_SubprocA.cc proc1/subprocs/subprocA/ProcA_SubprocA.h \
                      proc1/subprocs/subprocB/ProcA_SubprocB.cc proc1/subprocs/subprocB/ProcA_SubprocB.h \
                      proc2/subprocs/subprocA/ProcB_SubprocA.cc proc2/subprocs/subprocA/ProcB_SubprocA.h \
                      proc2/subprocs/subprocB/ProcB_SubprocB.cc proc2/subprocs/subprocB/ProcB_SubprocB.h

当我打字时

#libtoolize && autoreconf -i -f && ./configure && make
autoreconf -i -f && ./configure && make

我在 procs 中得到一个名为 libprocs.la 的文件,但我没有得到二进制文件。我也相当确定我的 Makefile 是错误的,因为我什至无法手动编译 ma​​in.cpp

是否可以在这里做我想做的事?还是我想得太多了,一开始我什至不需要图书馆?对我来说重要的部分是 ma​​in.cpp 只包含 Subprocs.h 然后包含所有包含的 subprocs。

提前致谢!

最佳答案

您介绍的内容有很多奇怪之处,还有一个明显的错误。你说,

I get a file called libprocs.la in procs but I don't get a binary. I am also fairly certain that my Makefiles are wrong since I can't even compile main.cpp by hand.

make 的输出肯定包含一条或多条解释构建失败原因的错误消息。除非你的 make 首先只访问 procs 子目录,我猜,在这种情况下 当然 你不会得到 主要内置。

但我有理由相信我可以猜到:您收到一个链接器错误,提示它找不到 ProcA_SubprocA::ProcA_SubprocA。这是因为虽然您的 Makefile 指定库应该构建(并且还安装),但它没有任何地方说它需要链接到 main。有一个非常简单的修复方法,我稍后会返回。

不过,首先让我们谈谈 libprocs.la.la 后缀对 Autotools 有意义,指定一个“libtool archive”,但您实际上是将它构建为一个普通的静态库。事实上,libtool 在这里并没有做任何对您特别有用的事情;我会通过从 configure.ac 中删除 LT_INIT 来转储它(并且不要再次运行 libtoolize,因为它会把它放回去)。然后,将要构建的库的名称更改为更标准的 libprocs.a

此外,由于该库似乎仅供正在构建的一个程序使用,因此将其单独包含在安装中是无益的。因此,与其在 lib_LIBRARIES 中指定它,不如将其指定为未安装的便利库:

noinst_LIBRARIES = libprocs.a

然后您需要对库的 SOURCES 变量进行相应的更改:

libprocs_la_SOURCES = ...

libprocs_a_SOURCES = ...

现在回到主程序,您需要告诉make(和Automake)库需要链接到程序main。关于如何做到这一点有一些变体,但我将建议一个相当简单的方法:

main_LDADD = procs/libprocs.a

在这些更改之后,重新运行 autoreconf(但不是 libtoolize)。但是请记住,只有在更改 Autotools 源之后才需要运行它——在您的案例中只是 Makefile.amconfigure.ac 文件。它是维护和开发包及其构建系统的详细信息。它不应该是构建或安装包的常规部分。

关于c++ - 使用 automake 将子目录编译到库中的最佳方法?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51037442/

相关文章:

c++ - libpq : get data type

c++ - 如何在派生类数据中使用基类进行比较

c++ - 我可以在 Mac 上使用 C++ 获得 -1 退出代码吗?

c - 使用 autotools 链接 gstreamer 插件中的外部库

linux - Autotools - 构建具有不同名称的静态和共享库

c++ - 与浏览器无关的 C++ DOM 接口(interface)

c++ - 如何在 automake 脚本中创建共享库 (.so)?

linux - 如何使用automake检查操作系统

linux - kdevelop 3.5 : configure: error: C++ compiler cannot create executables

选择 autoconf 和 automake 版本