疑惑点

C++是一门奇妙的语言。非常多时候你对底层不熟悉,非常难知道某些情况下的结果,以下是我不断积累的疑惑点,这里将其记录下来。

类的转换问题

代码:

class A{
public:
virtual void f()
{
cout << "A" << endl;
}
}; class B: public A{
public:
virtual void f()
{
cout << "B" << endl;
}
}; int _tmain(int argc, _TCHAR* argv[]){
A* pa = new A();
pa->f();
B* pb = (B*)pa;
pb->f(); delete pa,pb;
pa = new B();
pa->f(); //多态
pb = (B*)pa;
pb->f();
}

以下语句发生什么变化。

B* pb = (B*)pa;

解析:事实上什么变化也没有发生。还是输出A,不存在覆盖问题,pb指向pa原来地址。

虚函数中返回值类型不同。能覆盖吗?

答:不能覆盖。派生类重写基类的虚函数。返回值类型也必须同样。

基类默认构造派生类是否须要显示调用

例如以下代码报错吗?

class A{
public:
A(int a){}
}; class B: public A{
public:
B(){}
};

解释。在基类没有默认构造的情况下,派生类是须要显式调用。

所以上面代码编译不通过。

基类和派生类之间的转换问题

以下程序结果是什么:

class A{public: int m_a;};
class B{public: int m_b;};
class C:public A,public B{public: int m_c;}; int _tmain(int argc, _TCHAR* argv[]){
C* pc = new C;
pc->m_a = 1; pc->m_b = 2; pc->m_c = 3;
B* pb = dynamic_cast<B*>(pc);
A* pa = dynamic_cast<A*>(pc);
B pbb = *pc; //这里会发生什么? cout << pc << endl;
cout << pb << endl;
cout << pa << endl;
cout << &pbb << "sizeof:" << sizeof(pbb) << "m_b" << pbb.m_b << endl;
if (pc == pb)
cout << "equal" << endl;
else
cout << "not equal" << endl; if ((int)pc == (int)pa)
cout << "equal" << endl;
else
cout << "not equal" << endl;
}

解析:每一个语句解释例如以下,

1. A* pa = dynamic_cast<A*>(pc); 此时pa指向了子类A的那部分,地址值与pc同样。

2. B pbb = *pc; 这里会发生分割,调用了B类拷贝构造,将B的那部分分割到pbb的所在的栈空间中。

3. if (pc == pb) 这里会发生隐式类型转换,pc = (C*)pb

4. if ((int)pc == (int)pa) 尽管没有隐式类型转换,但地址同样。

dynamic_cast问题

以下代码中哪条语句会出现故障。

class A{
public:
virtual void foo(){cout << "A foo()" << endl; }
void pp(){ cout << "A pp()" << endl; }
}; class B:public A{
public:
void foo(){cout << "B foo()" << endl; }
void pp(){ cout << "B pp()" << endl; }
void funB(){ cout << "B::funB()" << endl;}
}; int _tmain(int argc, _TCHAR* argv[])
{
A a;
A *pa = &a;
(dynamic_cast<B*>(pa))->foo(); //语句1
(dynamic_cast<B*>(pa))->pp(); //语句2
(dynamic_cast<B*>(pa))->funB(); //语句3
system("pause");
return 0;
}

解析:语句1会出现故障。foo()是虚函数。编译器会依据对象的虚函数指针查找虚函数表,定位foo函数。

dynamic_cast不是强制类型转换,而是带有某种“咨询”性质的。假设不能转换,dynamic_cast会返回NULL。表示不成功。

上面3条语句相当于:

    B* bnull = NULL;
bnull->foo();
bnull->pp();
bnull->funB();

上面的转换时不成功的,所以返回的是NULL指针。又由于pp和funB函数未使用不论什么成员数据,也不是虚函数。不须要this指针,也不须要动态绑定,所以能够正常执行。

虚拟继承和带有虚函数的继承内存分布情况

以下代码输出结果是多少?

class A{
public:
virtual void foo(){};
private:
char ca[3];
}; class B:virtual public A{
public:
virtual void foo(){};
private:
char cb[3];
}; class C:virtual public B{
public:
virtual void foo(){};
private:
char cb[3];
}; int _tmain(int argc, _TCHAR* argv[])
{
cout << sizeof(A) << endl;
cout << sizeof(B) << endl;
cout << sizeof(C) << endl;
system("pause");
return 0;
}

解析:结果是8,16。24。

然而,以下代码输出结果又是多少?

去掉了虚拟继承

class A{
public:
virtual void foo(){};
private:
char ca[3];
}; class B:public A{
public:
virtual void foo(){};
private:
char cb[3];
}; class C:public B{
public:
virtual void foo(){};
private:
char cb[3];
}; int _tmain(int argc, _TCHAR* argv[])
{
cout << sizeof(A) << endl;
cout << sizeof(B) << endl;
cout << sizeof(C) << endl;
system("pause");
return 0;
}

解析:结果为8,12。6。

1. 带有虚函数的虚拟继承中,也分两种情况,1.派生类中定义了新的虚函数。而且部分重写了虚基类的虚函数,这个时候。派生类中有3个虚表指针,一个指向虚基类的虚函数指针,一个自己的虚表指针(实现多态),另一个基类自己虚表指针。

2.派生类所有重写虚基类虚函数,且没有新定义虚函数,这时候,派生类中含有两个虚表指针。另一个基类自己虚表指针,个指向虚基类的虚函数指针。

2. 基类不带虚函数的虚继承中,分两种情况,1.基类和派生类中都没有虚函数。这个时候,派生类仅仅会多加入一个虚表指针,指向虚基类的虚函数(尽管虚基类中没有虚函数)。2.基类中没有虚函数。派生类中有虚函数,则会生产两个虚表指针。一个指向自己虚函数的的虚表指针和一个指向虚基类的虚函数指针

3. 带虚函数的普通继承中,这个时候不论是基类还是派生类,仅仅要类中有虚函数。都会有且仅仅有一个虚函数指针。

再看以下代码:



验证了第一个的第一种情况。

在看以下:



这验证了第二个的另外一种情况。

另一个非常有趣的问题:



怎么没有内存对其了呢?这时假设再定义一个int就会有内存对齐了。假设没有就是以1字节对齐。

最后,总结下,虚拟继承和虚函数多态机制是分开的。虚拟继承会保留基类中的虚表指针,而且加入一个指向虚基类的虚拟指针,它并不会实现多态。

C/C++易错难点笔记01的更多相关文章

  1. 关于Verilog HDL的一些技巧、易错、易忘点(不定期更新)

    本文记录一些关于Verilog HDL的一些技巧.易错.易忘点等(主要是语法上),一方面是方便自己忘记语法时进行查阅翻看,另一方面是分享给大家,如果有错的话,希望大家能够评论指出. 关键词: ·技巧篇 ...

  2. 对象引用 方法传参 值传递 引用传递 易错点 MD

    Markdown版本笔记 我的GitHub首页 我的博客 我的微信 我的邮箱 MyAndroidBlogs baiqiantao baiqiantao bqt20094 baiqiantao@sina ...

  3. JavaGuide易错点总结

    基础知识易错点 1. object.equals("str") 容易报空指针异常,应使用"str".equals(object); 还可以使用JDK7引入的工具 ...

  4. Redis:学习笔记-01

    Redis:学习笔记-01 该部分内容,参考了 bilibili 上讲解 Redis 中,观看数最多的课程 Redis最新超详细版教程通俗易懂,来自 UP主 遇见狂神说 1. Redis入门 2.1 ...

  5. JavaScript易错知识点整理

    前言 本文是我学习JavaScript过程中收集与整理的一些易错知识点,将分别从变量作用域,类型比较,this指向,函数参数,闭包问题及对象拷贝与赋值这6个方面进行由浅入深的介绍和讲解,其中也涉及了一 ...

  6. JavaScript 易错知识点整理

    本文是我学习JavaScript过程中收集与整理的一些易错知识点,将分别从变量作用域,类型比较,this指向,函数参数,闭包问题及对象拷贝与赋值这6个方面进行由浅入深的介绍和讲解,其中也涉及了一些ES ...

  7. 软件测试之loadrunner学习笔记-01事务

    loadrunner学习笔记-01事务<转载至网络> 事务又称为Transaction,事务是一个点为了衡量某个action的性能,需要在开始和结束位置插入一个范围,定义这样一个事务. 作 ...

  8. 《30天自制操作系统》笔记(01)——hello bitzhuwei’s OS!

    <30天自制操作系统>笔记(01)——hello bitzhuwei's OS! 最初的OS代码 ; hello-os ; TAB=4 ORG 0x7c00 ; 指明程序的装载地址 ; 以 ...

  9. 《The Linux Command Line》 读书笔记01 基本命令介绍

    <The Linux Command Line> 读书笔记01 基本命令介绍 1. What is the Shell? The Shell is a program that takes ...

随机推荐

  1. 利用反射api查找一个类的具体信息

    讲到这个实例,首先介绍下本人,我是一个php程序猿.从事drupal开发2年多.能够说从实习開始就接触这个,至今没有换过.drupal给我的感觉是俩字"强大",今天写一个views ...

  2. c/c++中sleep()函数毫秒级的实现

    近期看到好多人在问.c/c++中的sleep函数是秒级的,能不能实现毫秒级的呢?当然非常easy.我的写法例如以下 #include <stdio.h> #include <sys/ ...

  3. Android菜鸟笔记- 获取未安装的APK图标、版本号、包名、名称、是否安装、安装、打开

    周末闲来无事,把Android的基础知识拿出来复习复习,今天主题是<获取未安装的APK图标.版本号.包名.名称.是否安装.跳转安装.打开> 一.获取APK图标 通常读取APK的图标能够用, ...

  4. j-link修复 write flash 一直无法点击

    write flash 一直无法点击 watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvcXFfMTcyNDI5NTc=/font/5a6L5L2T/fonts ...

  5. POJ 3905 Perfect Election(2-sat)

    POJ 3905 Perfect Election id=3905" target="_blank" style="">题目链接 思路:非常裸的 ...

  6. BroadcastReceiver类

    java.lang.Object    ↳ android.content.BroadcastReceiver 已知直接子类 AppWidgetProvider DeviceAdminReceiver ...

  7. 局域网网络性能測试方法HDtune 64K有缓存測速法,让你得知你的网络性能

    该方法能够有效測试出您的网络传输性能究竟有多高,该方法通用于有盘,无盘(系统虚拟盘) ,游戏虚拟盘,以及其它带有缓存功能的虚拟盘软件,可是由于每款软件的工作方式和原理都不仅同样,所以每款软件的測试结果 ...

  8. 绘图中的drawRect

    rect参数:代表的是当前view的bounds 1 为什么要在drawRect方法里面写绘图代码 因为只有在这个方法中才能获取到当前view相关的图形上下文对象 有了这个图形上写文对象后才能进行绘图 ...

  9. POJ 3150 循环矩阵的应用

    思路: 首先 先普及一个性质: 循环矩阵*循环矩阵=循环矩阵 由于此题是距离小于d的都加上一个数. 那么 构造矩阵的时候 我们发现 诶呦 这是个循环矩阵 看看数据范围 n^2log(k)可以过. 那就 ...

  10. CaffeNet用于Flickr Style数据集上的风格识别

    转自 http://blog.csdn.net/liumaolincycle/article/details/48501423 微调是基于已经学习好的模型的,通过修改结构,从已学习好的模型权重中继续训 ...