在 nullptr 上调用无状态类的非静态成员函数是否合法?
Is it legal to call a non-static member function of a stateless class on a nullptr?
请考虑以下代码:
int main()
{
struct EmptyStruct{
void nonstatic_mf() const { std::cout <<"EmptyStructn"; }
};
EmptyStruct *esptr = nullptr;
esptr->nonstatic_mf();
}
这是一个合法C++(它似乎确实适用于 gcc 和 clang(?
即使结构为空,它的大小也不为零。必须有一些内存充当它的存储。
不。这始终是 UB。如果不需要实例,请将其设为静态。静态函数仍然可以使用点.
语法进行调用。
为什么?因为不能取消引用空指针。调用等效于以下内容的成员函数:
EmptyStruct *esptr = nullptr;
(*esptr).nonstatic_mf();
如您所见,空指针是顺从的,即 UB。
标准对此有何规定?从 [class.mfct.non-static]/2:
如果为不属于 X 类型的对象或从 X 派生的类型的对象调用类 X 的非静态成员函数,则行为是未定义的。
空指针不指向EmptyStruct
的有效实例。仅此一项就足以使行为未定义
从 [expr.ref]/2:
对于第一个选项(点(,第一个表达式应为 glvalue。 对于第二个选项(箭头(,第一个表达式应是具有指针类型的 prvalue。 表达式
E1->E2
转换为等效形式(*(E1)).E2
;[expr.ref] 的其余部分将仅解决第一个选项(点(。
所以esptr->nonstatic_mf()
实际上等价于(*esptr).nonstatic_mf()
,而取消引用空指针是未定义的行为。
因此,有两种方式可以不定义此代码。
相关文章:
- 如果C++类在类方法中具有动态分配,但没有构造函数/析构函数或任何非静态成员,那么它仍然是POD类型吗
- 如何在C++中使用非静态成员函数作为回调函数
- 在作为静态成员包含在另一个类中的类的构造函数中使用 cout
- 静态成员函数使用相同的名称时出现模板类型名称错误
- 在 nullptr 上调用无状态类的非静态成员函数是否合法?
- 如何在友元函数中使用静态成员而不添加前缀 [类名]::
- C++构造函数和静态成员
- 为什么传递非静态成员函数会导致编译错误?
- 非静态成员失败的线程调用函数
- 处理类内的回调时,必须调用对非静态成员函数的引用
- 非静态成员函数的 decltype 格式不正确吗?
- 为什么 std::sort 找不到合适的(静态成员)函数重载?
- std::异步与非静态成员函数
- C++无效使用非静态成员函数?
- 指向重载静态成员的函数指针 - 在unique_ptr中用作自定义删除器
- 未使用的C++未优化的静态成员函数/变量
- C++:如何返回指向非静态成员函数的指针?
- 有没有一种方法可以使全局函数/静态成员函数一次可呼出
- 函数静态成员变量
- 隐藏模板化的辅助函数——静态成员或未命名的命名空间