C++ 使用带前向声明的成员函数

C++ using member functions with forward declaration

我也看到过类似的问题,但不太像我现在的困境。我在用别人的代码,他们的结构是这样的。

//db_manager.h
class db_manager
{
  class error;
  bool logError(error::def_enum::Value v, string msg);

  bool read(int id);
}

//db_manager.cpp
#include db_manager.h
bool logError(error::def_enum::Value v, string msg)
{
    return error::logError(v, msg);
}

bool read(int id)
{
    //do db access stuff
    return true;
}

//error.h
#include db_manager
class error
{
  bool read(int id);
}

//error.cpp
#include error.h
bool read(int id)
{
    return db_manager::read(id);
}

bool logError(error::def_enum::Value v, string msg)
{
    //do error service stuff
}

这是一个非常明显的简化,但希望它能说明问题。

我编译的时候,每当在db_manager.cpp中使用error时,我都会得到很多不完整的类型错误,而且我无法将error中的相关头文件包含到db_manager.cpp中,因为那时我必须将它添加到 db_managers cmake 依赖项,这意味着我必须将它列在 package.xml 中,然后它会因循环依赖而变得不安。我该如何解决这个问题?如果我可以在 db_manager 中使用 error 的成员而不使 error 成为依赖项,我想我会很好,但我就是不知道该怎么做。我在这里看到了许多其他前向声明问题,但对于所有这些问题,声明的 class 用法都不是很深。这里我使用的是 class 成员,而不是像其他问题那样只是声明一个 class 指针。

我绝对可以使用帮助,我只是看不出有任何合乎逻辑的方法来做到这一点而不完全废弃错误包并编写一个新包。

编辑:另外,我简化了这个,但也许我不应该这样做。 error 和 db_manager 在两个不同的包中。

首先:你的例子很糟糕。请提供一个最低工作示例。我明白你的问题是什么(循环依赖),但你的例子没有显示这个问题。这是您必须在架构层面上解决的问题。你无法在 CMake 中解决这个问题。

根据您显示的代码,您不需要在 error.h 中包含 db_manager.h,因为在 [=47] 的声明期间您没有使用 db_manager 中的任何内容=] 错误。您只需要将它包含在 error.cpp 中,因为您正在使用 db_manager 中的一种静态方法。这样你就没有任何循环依赖。

我在下面添加了一个最低限度的工作示例,它编译时没有任何错误。

error.h

#ifndef _ERROR_H_
#define _ERROR_H_

#include <string>

class Error
{
public:
  enum def_enum{ Val1, Val2};

  bool read(int id);
  static bool logError(def_enum v, std::string msg);
};

#endif /* _ERROR_H_ */

error.cpp

#include "error.h"
#include "db_manager.h"

bool Error::read(int id)
{
    return db_manager::read(id);
}

bool Error::logError(Error::def_enum v, std::string msg)
{
    //do error service stuff
    return true;
}

db_manager.h

#ifndef _DB_MANAGER_H_
#define _DB_MANAGER_H_

#include <string>
#include "error.h"

class db_manager
{
public:
  static bool logError(Error::def_enum v, std::string msg);
  static bool read(int id);
};

#endif /* _DB_MANAGER_H_ */

db_manager.cpp

#include "db_manager.h"

bool db_manager::logError(Error::def_enum v, std::string msg)
{
    return Error::logError(v, msg);
}

bool db_manager::read(int id)
{
    //do db access stuff
    return true;
}

main.cpp

#include "db_manager.h"
#include "error.h"

int main(){


  db_manager::read(1);
  db_manager::logError(Error::Val1, "Test");

  Error e;
  e.read(2);

  return 0;
}

CMakeLists.txt

project(db_manager)

add_executable(executable main.cpp db_manager.cpp error.cpp)