如何从C++内部类引用封闭实例
How to refer to enclosing instance from C++ inner class?
在C++中,对象通过this
引用自己。
但是,内部类的实例是如何引用其封闭类的实例的呢?
class Zoo
{
class Bear
{
void runAway()
{
EscapeService::helpEscapeFrom (
this, /* the Bear */
??? /* I need a pointer to the Bear's Zoo here */);
}
};
};
编辑
我对非静态内部类如何工作的理解是,Bear
可以访问其Zoo
的成员,因此它有一个指向Zoo
的隐式指针。在这种情况下,我不想访问成员;我正试图得到那个隐含的指针。
与Java不同,C++中的内部类没有对其封闭类实例的隐式引用。
你可以通过传递一个实例来模拟这一点,有两种方法:
传递到方法:
class Zoo
{
class Bear
{
void runAway( Zoo & zoo)
{
EscapeService::helpEscapeFrom (
this, /* the Bear */
zoo );
}
};
};
传递给构造函数:
class Zoo
{
class Bear
{
Bear( Zoo & zoo_ ) : zoo( zoo_ ) {}
void runAway()
{
EscapeService::helpEscapeFrom (
this, /* the Bear */
zoo );
}
Zoo & zoo;
};
};
内部类并不特殊,也没有任何到外部类的链接。如果你想访问外部类,那么就传递一个指针或引用,就像你对任何其他类一样。
您可以使用offsetof从内部类实例访问外部类实例
与指针/引用解决方案相比,这具有零开销
它有点脏,但它能完成任务
例如:
#include <cstddef>
struct enclosing {
struct inner {
enclosing& get_enclosing() {
return *(enclosing*)((char*)this - offsetof(enclosing, i));
}
void printX() {
std::cout << get_enclosing().x << 'n';
}
} i;
int x;
};
int main() {
enclosing e;
e.x = 5;
e.i.printX();
}
p.S.
offsetof对类型做了一些假设。在C++98中,类型必须是POD,而在C++11中,类型必须为"标准布局"
以下是参考资料:http://www.cplusplus.com/reference/cstddef/offsetof/
内部类可以访问外部类的所有成员,但它没有对父类实例的隐式引用。
要回答您修改后的问题:
不,您不能访问该隐式指针。我相信在Java中可以做到这一点,但在C++中不行。
您必须通过构造函数或其他函数显式传递外部类对象才能实现这一点。
从技术上讲,根据C++03标准(第11.8.1节),嵌套类对其封闭类没有特殊访问权限。
但也有这个标准缺陷:openstd.org/jtc1/sc22/wg21/docs/cwg_defaults.html#45不确定这是否已经关闭。
没有内置的机制来实现这一点。您需要通过构造函数或某种SetParent函数自己提供指针。
C++标准说(第11.8.1节[class.access.nest]):
嵌套类的成员没有对成员的特殊访问封闭类,也不到类或赋予友谊的功能到一个封闭类;通常的访问规则(第11条)服从封闭的成员类没有特殊访问权限嵌套类的成员;通常的访问规则(第11条)服从。
(我强调)。
这意味着嵌套类和封闭类之间没有特殊的关系。
在创建嵌套类的实例时,没有隐式创建的封闭类实例。必须手动完成。
- 对显式实例化的模板函数的未定义引用
- 重载运算符*以获取对另一个类的实例的引用
- 为什么我不能引用指向实例化对象的函数的指针?
- <Base> <Derived> 具有相同原始指针共享引用的 shared_ptr 和 shared_ptr 实例是否计数?
- 返回引用实例和非引用实例(return mystr & vs mystr)之间的区别是什么?
- 请参阅在 Visual Studio 2019 中捕获 std::exception 时对函数模板实例化消息的引用
- 将元组和整型实例合并到引用元组中
- C++ 为什么在定义的编译和链接之前引用外部实例的程序
- 是否允许使用初始值设定项列表将const数组引用实例化为构造函数参数
- 引用的静态强制转换强制模板实例化,其中不完整的类型很好
- 为什么不能在实例化对基类的引用的同时实例化指向派生类的指针?
- 非类型引用参数可以在运行时修改,这是否意味着模板可以在运行时实例化?
- 超出返回引用的单一实例生存期
- 反向函数不反转类实例化的引用字符串
- 通过引用将私有类实例传递到另一个类 C++ 的"get"函数中
- 多个实例保存对unique_ptr的引用
- 类模板实例化和通用引用
- 对从静态实例引用的类模板的静态成员的未定义引用
- 如何正确格式化类实例引用
- 如何在类构造函数中存储实例引用