下面3个小问题都是我认为C++ Beginner应该能够解答或辨别清楚的。希望我们能通过题目挖掘更多的信息,而不仅仅局限在解题。我最喜欢说的话:能力有限,所以作为抛砖引玉,希望共同讨论,指出错误。

另外,我都是碰到一个觉得有必要记录的问题,就写下来说说,所以每一篇内容可能不是单一主题。


1、先来看一道简单题目。有下面这个继承类:

class Person
{
public:
void Walk() //普通人的“走”
{
cout << "Person::Walk I am an Ordinary People." << endl;
};
}; class Student : public Person
{
public:
void Walk() //学生的“走”
{
cout << "Student::Walk I am a student." << endl;
};
};

你没看错Walk()是非虚函数。请解释下面代码:

Student s;
Person* pp = &s;
pp->Walk(); Student* ps= &s;
ps->Walk();

结果是这样的:

分析:Walk()是非虚函数,被静态绑定所限制,所以pp、ps是什么类型就决定了调用的版本。这里,我还要说明的一点是:明白接口继承和实现继承。声明一个non-virtual函数的目的是为了令derived class继承函数的接口及一份强制性实现。所以,绝不要重新定义继承而来的non-virtual函数


 2、下面这个问题实质上也是静态绑定与动态绑定的问题,但看起来不那么明显。

class Shape
{
public:
enum ShapeColor{Red, Green, Blue}; //形状颜色 virtual void Draw(ShapeColor color = Red) const = ;
}; class Circle : public Shape
{
public:
virtual void Draw(ShapeColor color) const
{
cout << "I am Circle::Draw. ";
cout << "My color = " << color << endl;
}
}; class Rectangle : public Shape
{
public:
virtual void Draw(ShapeColor color = Green) const //缺省的参数值被更改了
{
cout << "I am Rectangle::Draw. ";
cout << "My color = " << color << endl;
}
};

我主要想说两个问题。

(1)当你下面这样调用时,请解释会发生什么情况。

Circle cr;            //(1) 编译不通过
cr.Draw(); Shape *ps = &cr; //(2)
ps->Draw();

没错,(1)通过对象调用而不指定参数是错误的,而(2)的结果是这样的:color = 0代表Red这你应该是知道的。

分析:通过对象调用是静态绑定,一定要指定参数值,因为静态绑定这个函数不从base class继承缺省参数值。动态绑定却可以从base class继承参数值。注意,这里我就不强调动态绑定和静态绑定的概念了,但下面这个一定是静态绑定:

Circle cr;
Circle *ps = &cr; //这还是静态绑定,静态类型Circle *,编译不通过
ps->Draw();

(2)第二个我想说的问题,请解释下面的调用结果。

Shape* ps1 = new Rectangle;
ps1->Draw(); Shape* ps2 = new Circle;
ps2->Draw();

是这样令人可喜的结果:

你是说,你在Rectangle中已经将Draw的缺省值改为1(Green)了,怎么没效果?

分析:Rectangle::Draw的缺省参数值为GREEN,但ps2的静态类型为Shape*,所以此调用的缺省参数值来自Shape class。

如果你非要让Rectangle::Draw的参数有所改变,可以这样调用(提供参数):

Shape* ps4 = new Rectangle;
ps4->Draw(Shape::Green); Shape* ps5 = new Circle;
ps5->Draw(Shape::Green);

这个问题是想提醒你:virtual函数是动态绑定,缺省参数值是静态绑定。所以,不应该重新定义这个缺省参数值。


3、多重继承为什么会含有多个虚表指针而不是一个?

这道题是我看一位同学面试经验时,面试官提的,我试着回答一下,不知道在不在点子上,还请补充和指正。

答:多重继承下,因为编译器对一个derived class实现了n-1个虚表,n表示上一层base class的个数,当然假设每个base class都有至少有一个virtual函数,否则编译器是不会为其添加vptr和vtbl了。所以说有多少个虚表,自然就有多少个指针指向,而不是一个。

这样说我不知道合理不合理,可能面试官要问的点是“为什么需要多个虚表?一个虚表行不行?”

这个属于编译器厂商做的事情,标准并未规范。C++的父亲就做出过这样的一款编译器原型,通过增大vtbl的体积,每个slot上不只有一个指针,还有一个offset,用来调整this指针的指向。

这样做的弊端是:所有vtbl中的虚函数指针都包含这样一个offset,并且假设不需要调整this指向,调用时还是要做offset的加法操作,尽管offset此时为0。另外,vtbl中每个slot体积的膨胀。这些都是效率问题。

实际上,用来调整this的指向用的比较多的是trunk技术,必须以汇编编写才能获得高效率。另外,sun编译器就是把多个虚表连锁为1个,每个表格中含有下一个表格的指针(通过offset方式),这样就需要一个指针就好了。

理解能力有限,不知道问的是不是这么一回事?


目录:

C++中几个值得分析的小问题(1)

C++中几个值得分析的小问题(2)

C++中几个值得分析的小问题(1)的更多相关文章

  1. C++中几个值得分析的小问题(2)

    下面有3个小问题,作为C++ Beginner你一定要知道错在哪里了. 1.派生类到基类的引用或指针转换一定“完美”存在? 一般情况,你很可能会认为:派生类对象的引用或指针转换为基类对象的引用或指针是 ...

  2. SPSS缺失值得分析处理

    SPSS缺失值得分析处理 在资料收集的过程中,由于各种原因可能导致数据收集不全,就会产生缺失值,且这种情况往往无法避免.如果缺失值处理不当,就会导致分析结果精度降低,出现偏倚甚至是错误的理论,因此缺失 ...

  3. AI框架中图层IR的分析

    摘要:本文重点分析一下AI框架对IR有什么特殊的需求.业界有什么样的方案以及MindSpore的一些思考. 本文分享自华为云社区<MindSpore技术专栏 | AI框架中图层IR的分析> ...

  4. 容器中的诊断与分析4——live diagnosis——LTTng

    官网地址 LTTng 简介&使用实战 使用LTTng链接内核和用户空间应用程序追踪 简介: LTTng: (Linux Trace Toolkit Next Generation),它是用于跟 ...

  5. 从集合中查找最值得方法——max(),min(),nlargest(),nsmallest()

    从集合中查找最值得方法有很多,常用的方法有max(),min(),nlargest(),nsmallest()等. 一.max()和min() 1.1 入门用法 直接使用max(),min(),返回可 ...

  6. 「Python-Django」Django中使用数据库的 9 个小技巧

    Django 中使用数据库的 9 个小技巧. 1. 过滤器聚合 在 Django 2.0 之前,如果你想得到“用户总数”.“活跃用户总数”等信息时,你不得不使用条件表达式. Django 2.0 中, ...

  7. CSS中cursor属性给标签加上小手形状

    HTML/CSS 2012-08-10 CSS,标签 我们发现a标签在网页中有一个值得注意的地方,即鼠标移到a标签上光标会变成一只小手的图标,移出a标签后又恢复为默认箭头. 如今,JS在网页中的功能越 ...

  8. LoadRunner中对图表的分析说明

    LoadRunner中对图表的分析说明 (一)在Vusers(虚拟用户状态)中 1.Running Vusers(负载过程中的虚拟用户运行情况) 说明——系统形成负载的过程,随着时间的推移,虚拟用户数 ...

  9. [转]在ASP.NET开发中容易忽略的2个小问题 Cookie乱码存取异常 和 iframe弹框的login跳转

    本文转自:http://www.cnblogs.com/outtamyhead/p/3642729.html 本文地址:http://www.cnblogs.com/outtamyhead/p/364 ...

随机推荐

  1. hadoop 一些命令

    关闭访问墙  service iptables stop hadoop dfs -mkdir input hadoop dfs -copyFromLocal conf/* input hadoop j ...

  2. xlrd的使用操作

    # _*_ coding:utf-8 _*_ #---------------------------------------------------------------------------- ...

  3. PHP分页及原理

    在看本文之前,请确保你已掌握了PHP的一些知识以及MYSQL的查询操作基础哦. 作为一个Web程序,经常要和不计其数的数据打交道,比如会员的数据,文章数据,假如只有几十个会员那很好办,在一页显示就可以 ...

  4. JS正则表达式从入门到入土(6)—— 贪婪模式与非贪婪模式

    贪婪模式 之前说了正则的量词,但是量词会带来一个到底该匹配哪个的问题. 如下正则表达式: \d{3,6} 这个正则表达式是匹配3到6个数字,但是当这个正则表达式被用来匹配12345678这个字符串,到 ...

  5. 架构私用Nuget服务器

    1.新建一个空的asp.net站点 2.通过nuget引用 Nuget.Server程序集,引用后项目会多出一些文件.修改web.config 里的apikey为你要上传包时用的apikey,我的为: ...

  6. javascript 判断数据类型的几种方法

    javascript 判断数据类型的几种方法一.typeof 直接返回数据类型字段,但是无法判断数组.null.对象 typeof 1 "number" typeof NaN &q ...

  7. spring boot 集成kafka (多线程,消费者使用kafka的原生api实现,因为@KakfkaListener修改groupId无效)

    application-test.properties #kafka kafka.consumer.zookeeper.connect=*:2181 kafka.consumer.servers=*: ...

  8. Linux 动态链接库包含静态链接库的方法

    今天老司机们在讨论一个编译问题  A是一个静态库  C是一个动态库  B是运行程序,能不能将A打包到C 然后B只需要链接C 就可以了. 这个问题我以前在出来zlib库版本冲突的时候有点印象,所以写了个 ...

  9. layui和bootstrap 对比

    layui和bootstrap 对比 这两个都属于UI渲染框架. layui是国人开发的一套框架,2016年出来的,现在已更新到2.X版本了.比较新,轻量级,样式简单好看. bootstrap 相对来 ...

  10. [Linux]pycharm在Linux环境下安装

    之前转载了一个在Windows环境下pycharm专业破解的安装的文章,今天为了在Linux环境下安装使用odoo10,所以尝试在Linux环境下安装pycharm专业破解版看看. windows下安 ...