jjzjj

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

coder 2023-11-17 原文

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

这个例子取自不同的 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++ - 需要理解语句 "Accessability is checked statically and not dynamically in C++"的更多相关文章

  1. ruby-on-rails - rails : "missing partial" when calling 'render' in RSpec test - 2

    我正在尝试测试是否存在表单。我是Rails新手。我的new.html.erb_spec.rb文件的内容是:require'spec_helper'describe"messages/new.html.erb"doit"shouldrendertheform"dorender'/messages/new.html.erb'reponse.shouldhave_form_putting_to(@message)with_submit_buttonendendView本身,new.html.erb,有代码:当我运行rspec时,它失败了:1)messages/new.html.erbshou

  2. ruby - 我需要将 Bundler 本身添加到 Gemfile 中吗? - 2

    当我使用Bundler时,是否需要在我的Gemfile中将其列为依赖项?毕竟,我的代码中有些地方需要它。例如,当我进行Bundler设置时:require"bundler/setup" 最佳答案 没有。您可以尝试,但首先您必须用鞋带将自己抬离地面。 关于ruby-我需要将Bundler本身添加到Gemfile中吗?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/4758609/

  3. ruby-on-rails - 由于 "wkhtmltopdf",PDFKIT 显然无法正常工作 - 2

    我在从html页面生成PDF时遇到问题。我正在使用PDFkit。在安装它的过程中,我注意到我需要wkhtmltopdf。所以我也安装了它。我做了PDFkit的文档所说的一切......现在我在尝试加载PDF时遇到了这个错误。这里是错误:commandfailed:"/usr/local/bin/wkhtmltopdf""--margin-right""0.75in""--page-size""Letter""--margin-top""0.75in""--margin-bottom""0.75in""--encoding""UTF-8""--margin-left""0.75in""-

  4. ruby - 检查 "command"的输出应该包含 NilClass 的意外崩溃 - 2

    为了将Cucumber用于命令行脚本,我按照提供的说明安装了arubagem。它在我的Gemfile中,我可以验证是否安装了正确的版本并且我已经包含了require'aruba/cucumber'在'features/env.rb'中为了确保它能正常工作,我写了以下场景:@announceScenario:Testingcucumber/arubaGivenablankslateThentheoutputfrom"ls-la"shouldcontain"drw"假设事情应该失败。它确实失败了,但失败的原因是错误的:@announceScenario:Testingcucumber/ar

  5. ruby-on-rails - 如何优雅地重启 thin + nginx? - 2

    我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server

  6. ruby - rspec 需要 .rspec 文件中的 spec_helper - 2

    我注意到像bundler这样的项目在每个specfile中执行requirespec_helper我还注意到rspec使用选项--require,它允许您在引导rspec时要求一个文件。您还可以将其添加到.rspec文件中,因此只要您运行不带参数的rspec就会添加它。使用上述方法有什么缺点可以解释为什么像bundler这样的项目选择在每个规范文件中都需要spec_helper吗? 最佳答案 我不在Bundler上工作,所以我不能直接谈论他们的做法。并非所有项目都checkin.rspec文件。原因是这个文件,通常按照当前的惯例,只

  7. ruby - 如何在 Lion 上安装 Xcode 4.6,需要用 RVM 升级 ruby - 2

    我实际上是在尝试使用RVM在我的OSX10.7.5上更新ruby,并在输入以下命令后:rvminstallruby我得到了以下回复:Searchingforbinaryrubies,thismighttakesometime.Checkingrequirementsforosx.Installingrequirementsforosx.Updatingsystem.......Errorrunning'requirements_osx_brew_update_systemruby-2.0.0-p247',pleaseread/Users/username/.rvm/log/138121

  8. ruby-on-rails - 迷你测试错误 : "NameError: uninitialized constant" - 2

    我遵循MichaelHartl的“RubyonRails教程:学习Web开发”,并创建了检查用户名和电子邮件长度有效性的测试(名称最多50个字符,电子邮件最多255个字符)。test/helpers/application_helper_test.rb的内容是:require'test_helper'classApplicationHelperTest在运行bundleexecraketest时,所有测试都通过了,但我看到以下消息在最后被标记为错误:ERROR["test_full_title_helper",ApplicationHelperTest,1.820016791]test

  9. ruby-on-rails - 相关表上的范围为 "WHERE ... LIKE" - 2

    我正在尝试从Postgresql表(table1)中获取数据,该表由另一个相关表(property)的字段(table2)过滤。在纯SQL中,我会这样编写查询:SELECT*FROMtable1JOINtable2USING(table2_id)WHEREtable2.propertyLIKE'query%'这工作正常:scope:my_scope,->(query){includes(:table2).where("table2.property":query)}但我真正需要的是使用LIKE运算符进行过滤,而不是严格相等。然而,这是行不通的:scope:my_scope,->(que

  10. 使用 ACL 调用 upload_file 时出现 Ruby S3 "Access Denied"错误 - 2

    我正在尝试编写一个将文件上传到AWS并公开该文件的Ruby脚本。我做了以下事情:s3=Aws::S3::Resource.new(credentials:Aws::Credentials.new(KEY,SECRET),region:'us-west-2')obj=s3.bucket('stg-db').object('key')obj.upload_file(filename)这似乎工作正常,除了该文件不是公开可用的,而且我无法获得它的公共(public)URL。但是当我登录到S3时,我可以正常查看我的文件。为了使其公开可用,我将最后一行更改为obj.upload_file(file

随机推荐