仅头文件库循环依赖

Header-only library circular dependency

本文关键字:循环 依赖 文件      更新时间:2023-10-16

我正试图围绕外部C API创建一个仅头文件的c++库。C API使用void *指针作为句柄。思路如下:

// resource.hpp
class Resource {
public:
    // RAII constructor, destructor, etc.
    // ...
    void do_something(const Handle & h) {
        do_something_impl( (void *) h);
    }
};
// handle.hpp
class Handle
{
public:
    Handle(size_t n, const Resource & res)
        : p_(res.allocate(n)), res_(res) {}
    // cast operation
    operator void *() const { return p_; }
private:
    void * p_;
    Resource & res_;
};

这里的问题是(a)句柄必须保持对资源的引用,并且(b)资源需要能够将句柄强制转换为void *。不幸的是,这会导致循环依赖。

关于如何重构这个有什么想法吗?

注意:答案不是简单地"包含xxx.hpp"或前向声明一个类。这需要以某种方式重组,我只是不太明白该怎么做。

class Handle作为前向声明添加到资源文件的顶部不起作用,因为(void *)转换是Resource仍然无法看到的Handle定义的一部分。同样,将强制转换为void * ptr()成员函数也会导致同样的问题。

将函数定义移动到.cpp文件中也不是答案——它需要是头文件。

好吧,这是模板的拯救(再次!):

// resource.hpp
class Resource;
template<typename TResource> class Handle;
class Resource {
public:
    // RAII constructor, destructor, etc.
    // ...
    void do_something(const Handle<Resource> & h) {
        do_something_impl( (void *) h);
    }
};
// handle.hpp
template<class TResource>
class Handle {
public:
    Handle(size_t n, const TResource & res)
        : p_(res.allocate(n)), res_(res) {}
    // cast operation
    operator void *() const { return p_; }
private:
    void * p_;
    TResource & res_;
};

不要将头文件包含在彼此之间,而是使用向前声明类。这样,它们就可以用作引用或指针。

在resource.hpp:

class Handle;
class Resource {
public:
    // RAII constructor, destructor, etc.
    // ...
    void do_something(const Handle & h) {
        do_something_impl( (void *) h);
    }
};

And

class Resource;
class Handle
{
public:
    Handle(size_t n, const Resource & res)
        : p_(res.allocate(n)), res_(res) {}
    // cast operation
    operator void *() const { return p_; }
private:
    void * p_;
    Resource & res_;
};

由于在do_something函数中使用void*类型转换操作符,因此必须将该实现移动到单独的源文件中。在这个源文件中,你可以包括两个头文件,这样所有的函数都可以被访问。