将前向声明类的成员函数声明为友元

Declare a member-function of a forward-declared class as friend

本文关键字:声明 函数 友元 成员      更新时间:2023-10-16

是否可以将前向声明类的成员函数声明为友元?我正在尝试做以下事情:

class BigComplicatedClass;
class Storage {
   int data_;
public:
   int data() { return data_; }
   // OK, but provides too broad access:
   friend class BigComplicatedClass;
   // ERROR "invalid use of incomplete type":
   friend void BigComplicatedClass::ModifyStorage(); 
};

因此,目标是(i)将友元声明限制为单个方法,以及(ii)不包括复杂类的定义,以减少编译时间。

一种方法可能是添加一个充当中介的类:

// In Storage.h:
class BigComplicatedClass_Helper;
class Storage {
    // (...)
    friend class BigComplicatedClass_Helper;
};
// In BigComplicatedClass.h:
class BigComplicatedClass_Helper {
     static int &AccessData(Storage &storage) { return storage.data_; }
     friend void BigComplicatedClass::ModifyStorage();
};

然而,这似乎有点笨拙。。。所以我认为一定有更好的解决方案!

正如@Ben所说,这是不可能的,但您可以通过"passkey"授予对该成员函数的特定访问权限。它的工作方式有点像中间助手类,但更清晰:

// Storage.h
// forward declare the passkey
class StorageDataKey;
class Storage {
   int data_;
public:
   int data() { return data_; }
   // only functions that can pass the key to this function have access
   // and get the data as a reference
   int& data(StorageDataKey const&){ return data_; }
};
// BigComplicatedClass.cpp
#include "BigComplicatedClass.h"
#include "Storage.h"
// define the passkey
class StorageDataKey{
  StorageDataKey(){} // default ctor private
  StorageDataKey(const StorageDataKey&){} // copy ctor private
  // grant access to one method
  friend void BigComplicatedClass::ModifyStorage();
};
void BigComplicatedClass::ModifyStorage(){
  int& data = storage_.data(StorageDataKey());
  // ...
}

不,在声明单个成员函数之前,不能将它们声明为友元。你只能和全班同学交朋友。

这里可能相关,也可能不相关,但提醒我们自己,在类和对象的范围之外,还有一个狂野的世界,函数可以自由漫游。

例如,我最近需要从一个基于他人代码端口的全局异常处理程序中关闭一个(单例全局静态)系统错误日志。错误日志的正常包含文件与异常处理程序代码冲突,因为两者都希望包含"windows.h",原因我没有研究。当这个问题和其他问题说服我,我不能对ErrorLog类的成员函数进行正向声明时,我所做的是将必要的函数包装成一个全局范围函数,如下所示:

void WriteUrgentMessageToErrorLog( const char * message )
{
  ErrorLog::LogSimpleMessage( message );
  ErrorLog::FlushAccumulatedMessagesToDisk();
}

有些人非常注重不惜一切代价保持阶级结构的完整性。。。很少承认使用这些类的应用程序不可避免地建立在缺乏这种结构的基础上。但它就在那里,而且使用得当,它有它的位置。

考虑到这个问题的年龄,我没有深入研究它在这里的相关性。我想分享的是这样一种观点,即有时像这样简单的包装机制是一种更干净、更容易理解的替代方案,而不是一种更微妙、更聪明的东西。微妙和聪明往往会在以后的某个日期被不完全理解它的人要求添加到它中。在你意识到它之前,你有一个错误。。。