将 pimpl 样式 class 定义编译并链接到库中 (C++)

Compiling and linking pimpl style class definition into a library (C++)

我有一个我使用的 class,我试图重写它,以便它使用 pimpl 习语作为练习。但是,我现在在尝试编译和 link 程序时遇到了麻烦。我有一个主文件夹,其中包含一个名为 data 的文件夹,其中声明了 class Data 的 pimpl 实现。此文件夹再次包含文件夹 dataimpl,其中实施了 class DataImpl。这棵树是

├── data
│   ├── dataimpl
│   ├── lib

class Data 给出为

#ifndef INCLUDED_DATA_
#define INCLUDED_DATA_

#include "dataimpl/dataimpl.h"

class Data
{
    DataImpl * pimpl; // Only internal variable

public:
    Data();
    bool read();
    void display() const;
};

#endif

我可以创建这样的可执行文件

g++-6 main.cc data/*.cc data/dataimpl/*.cc -std=c++14

而且效果很好。但是,我想先在文件夹 data/lib 中为 Data class 创建一个库文件。为此,我在文件夹 data 中使用命令

g++-6 -c *.cc
ar -rsv lib/libdata.a
ranlib lib/libdata.a 

然后我将 main.cc 的目标文件编译为

g++-6 -c main.cc 

现在我想通过 linking as

把整个东西放在一起
g++-6 -o exec main.o data/lib/libdata.a data/dataimpl/*.cc

我收到错误消息

main.o: In function `main':
main.cc:(.text+0x26): undefined reference to `Data::Data()'
main.cc:(.text+0x3a): undefined reference to `Data::read()'
main.cc:(.text+0x7d): undefined reference to `Data::display() const'
collect2: error: ld returned 1 exit status

我无法理解为什么 Data 函数在 link 时将库文件添加到搜索路径时会给出未定义的引用。如果找不到 DataImpl 函数对我来说是有意义的,但我也将它们添加到搜索路径中(实现在 data/dataimpl 中并具有 .cc 扩展名)。

你的 ar 命令

ar -rsv lib/libdata.a

应该是

ar -rsv lib/libdata.a *.o

否则您创建了一个空存档。顺便说一句,您应该熟悉显示符号的 linux 命令 'nm'。