c++ - 需要理解语句 "Accessability is checked statically and not dynamically in C++"

标签 c++ inheritance access-specifier

我对静态或动态检查访问说明符感到困惑。 据说不会动态检查访问说明符。这是什么意思?

这个例子取自不同的 posts所以。考虑这个例子

示例 A:

class Base 
{
public:
    virtual void Message() = 0;
};

class Intermediate : public Base 
{
   //Is Message method virtual here too ? is it private or public ?
};

class Final : public Intermediate {
    void Message() {
        cout << "Hello World!" << endl;
    }
};

 Final final;

现在假设我做这样的事情

Final* finalPtr = &final;
finalPtr->Message();

上面的方法行不通,我的理解是在 Final 类中,消息方法是私有(private)的。那是对的吗 ?如果是这样,为什么这行得通?

中的方法
  Intermediate* finalPtr = &final; // or Base* finalPtr = &final;
  finalPtr->Message();

之所以上面的代码可以工作,是因为基类指针是用派生类实例化的。如果是这样,为什么对 Message() 的调用有效。 SO 上的帖子指出,由于继承是公共(public)的,因此它将作为公共(public)函数继承?另一方面,类中的函数本身具有私有(private)访问说明符(因为默认情况下它在类中是私有(private)的)。 我在这里很困惑,如果有人能澄清这一点,我将不胜感激。如果用派生类实例化基类,这样说是否正确。那么基类方法的访问说明符优先于派生类的访问说明符?

更新:

我还注意到,如果我将 IntermediateFinal 代码更改为以下内容

class Intermediate : public Base 
{
public: //Incase this public was absent then the following example wont work
    void Message() {
        cout << "Hello World! Intermediate" << endl;
    }
};

class Final : public Intermediate {
    void Message() {
        cout << "Hello World! final" << endl;
    }
};

并这样使用它

 Intermediate* i = new Final();
    i->Message();

然后为了得到输出 "Hello World!final" 有必要将接口(interface)方法标记为 public。这是为什么 ? Message 方法被继承为 public 。为什么我现在需要将其标记为公开?

最佳答案

您引用中的语句意味着访问检查基于您已将 . 运算符应用于(或等效于 -> 的表达式的静态类型到 *.).

如果 T 是一个类型,则:

 T *t = something....;
 t->foo();

访问检查是针对名称 T::foo,即使指针实际上指向从 foo 派生的类的对象。

换句话说,访问检查必须能够在编译时执行。不存在“运行时可访问性故障”这样的事情。


在您的代码示例中,您有:

Intermediate* finalPtr = something.....;
finalPtr->Message();

正在查找的名称是 Intermediate::MessageIntermediate 类将 Message 作为 public 函数,因此此检查成功。

您的评论表明您可能不确定继承函数的可访问性。推导分为三种类型(有点容易混淆,它们也被称为privateprotectedpublic)。 public 继承是最常见的;这意味着基类的 public 成员也是派生类的 public 成员。

由于 Base::Message 是公开的,而 Intermediate 是从 Base 公开派生的,那么 Intermediate::Message也是公开的。

需要明确的是,名称 具有访问权限。 Base::MessageIntermediate::Message 是两个不同的名称,但它们都命名相同的函数。

Final 中,它在其 private 部分声明了名称 Message,这意味着 Final::Message 是私有(private)的。尽管继承是公开的,但名称 Final::Message 的新声明隐藏了从 Base 继承的名称。


回答“更新”。当代码为:

class Intermediate : public Base 
{
public: //Incase this public was absent then the following example wont work
    void Message() {
        cout << "Hello World! Intermediate" << endl;
    }
};

您已将 Intermediate::Message 声明为公开的。因此,您可以通过 Intermediate * 类型的指针调用该函数。这就是“公众”的意思。如果您将其设置为 private,则无法调用它。

我不太明白您所说的“Message 方法被继承为 public 。为什么我现在需要将其标记为 public”是什么意思。当您在 class Intermediate 中编写 void Message () 时,它声明了一个新函数。该函数的名称隐藏了继承的名称。

关于c++ - 需要理解语句 "Accessability is checked statically and not dynamically in C++",我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25944999/

相关文章:

c++ - 'void*' 到 'char"分配中的不兼容类型 ------ 在 malloc 中

Python 元类(abc 模块)继承与嵌套类

ruby - 了解 Ruby 中的私有(private)方法

c++ - 基类的 "default"访问说明符差异的基本原理

c++ - 一个带有传递参数的程序,为什么这个程序有错误的答案?

c++ - gotoxy() 函数在 Visual Studio 中不起作用

C++ API 设计方法

c++ - 在不知道派生类的情况下从父类调用派生类函数

c# - 泛型继承方法

c++ - 在 C++ 中编译类的代码时,控制流的顺序是什么?