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

Best way to compile subdirectories into libraries with automake?

我有以下文件

├── 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 中的所有内容创建一个库,以便它可以在 main.cpp 中使用,看起来像这样:

main.cpp

#include "Subprocs.h"

int main(){

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

    return 0;
}

我想在 Subprocs.h 中包含子进程的所有包含,因此它可以用作某种接口:

Subprocs.h

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

我的configure.ac看起来像这样:

configure.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的像这样

Makefile.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 是错误的,因为我什至无法手动编译 main.cpp

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

提前致谢!

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

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 子目录,否则我猜,在这种情况下 当然 您将无法构建 main .

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

不过,首先让我们谈谈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 文件。它是维护和开发包及其构建系统的详细信息。它不应该是构建或安装包的常规部分。