友元函数问题

friend function issues

本文关键字:问题 函数 友元      更新时间:2023-10-16

我在C++中使用友元函数时遇到了一些困难,但我怀疑这更像是我在预处理器指令和 #include 方面遇到的问题的症状。

这是我正在做的事情的一个愚蠢的例子。五个文件:bobby.h、bobby.cpp、billy.h、billy.cpp 和 main.cpp。Billy有一个名为ReceiveMoney的受保护函数。Bobby有一个名为bank的函数,它调用Billy's ReceiveMoney。也就是说,每次鲍比去银行时,他都会和比利一起分钱。

比利·

#ifndef BILLY_H
#define BILLY_H
#include "bobby.h"
class Billy
{
friend void Bobby::Bank(int, Billy &);
public:
    Billy();
protected:
    void ReceiveMoney(int inc);
private:
    int money;
};
#endif

比利.cpp

#include "billy.h"
Billy::Billy()
{
    money = 0;
}
void Billy::ReceiveMoney(int inc)
{
    money+=inc;
}

鲍比·

#ifndef BOBBY_H
#define BOBBY_H
#include "billy.h"
class Bobby
{
public:
    Bobby();
    void Bank(int amount, Billy & b);
protected:
    int money;
};
#endif

鲍比.cpp

#include "bobby.h"
Bobby::Bobby()
{
    money = 0;
}
void Bobby::Bank(int amount, Billy & b)
{
    b.ReceiveMoney(amount/2);
}

主.cpp

#include "billy.h"
#include "bobby.h"
int main()
{
    Bobby bo;
    Billy bi;
    bo.Bank(150, bi);
    return 0;
}

我收到大量错误,通常错误 C2653:"Bobby":不是类或命名空间名称或错误 C2653:"Billy":不是类或命名空间名称

我在VS0中的一个空控制台项目中执行此操作

您有一个头文件的循环依赖关系
billy.h包括bobby.h,而bobby.h包括billy.h
显然,由于这种循环依赖关系,编译器无法辨认出类型。

最好的解决方案是重新考虑您的设计并避免循环依赖或
使用前向声明中断循环依赖关系。

只需在bobby.h中向前声明类Billy

//#include "billy.h"     <-----  You don't need this include 
class Billy;             <-----  Just Forward Declaration should suffice

您可以在此处使用前向声明,因为声明接受/返回不完整类型的函数或方法,在本例中Billy是编译器的不完整类型。

你的 #include 有一个循环,你不能那样做。您必须在 billy.h 中使用 Bobby 的前向声明。比如class Bobby;.即使这样,你也无法声明 friend 函数。

唯一真正的解决方案是 避免对friend的需求 . 事实上,ReceiveMoney应该是公开的:如果Bobby代表一个帐户之类的东西,这是合乎逻辑的。

friend的约束使得它只对解决类的内部行为有用(例如,集合和实现它们的节点)。

由于循环依赖关系,没有一个类是完全定义的。 因此存在大量错误。如果可能,请更改您的设计并继承或仅包含必要的内容。正如Als所提到的,前瞻声明可能是一种选择。循环依赖关系主要是由于设计错误引起的。