继承与接口之覆盖之一

题:以下代码的输出结果是什么?【中国著名门户网站W公司2007年9月校园招聘面试题】

// P123_example1.cpp : Defines the entry point for the console application.
//

#include "stdafx.h"
#include <iostream>

class A
{
protected:
	int m_data;
public:
	A(int data = 0)
	{
		m_data = data;
	}
	int GetData()
	{
		return doGetData();
	}
	virtual int doGetData()
	{
		return m_data;
	}
};

class B : public A
{
protected:
	int m_data;
public:
	B(int data = 1)
	{
		m_data = data;
	}
	int doGetData()
	{
		return m_data;
	}
};

class C : public B
{
protected:
	int m_data;
public:
	C(int data = 2)
	{
		m_data = data;
	}
};


int _tmain(int argc, _TCHAR* argv[])
{
	C c(10);
	std::cout<<c.GetData()<<std::endl;
	std::cout<<c.A::GetData()<<std::endl;
	std::cout<<c.B::GetData()<<std::endl;
	std::cout<<c.C::GetData()<<std::endl;
	std::cout<<c.doGetData()<<std::endl;
	std::cout<<c.A::doGetData()<<std::endl;
	std::cout<<c.B::doGetData()<<std::endl;
	std::cout<<c.C::doGetData()<<std::endl;
	return 0;
}
解析:

构造函数从最初始的基类开始构造,各个类的同名变量没有形成覆盖,都是单独的变量。理解这两个重要的C++特性后解决这个问题就比较轻松了。下面我们详解这几条输出语句。

std::cout<<c.GetData()<<std::endl;
本来是要调用C类的GetData(),C中未定义,故调用B中的,但是B中也未定义,故调用A中的GetData(),因为A中的doGetData()是虚函数,所以调用B类中的doGetData(),而B类的doGetData()返回B::m_data,故输出1。

std::cout<<c.A::GetData()<<std::endl;
因为A中的doGetData()是虚函数,又因为C类中未重定义该接口,所以调用B类中的doGetData(),而B类的doGetData()返回B::m_data,故输出1。

std::cout<<c.B::GetData()<<std::endl;
肯定返回1了。

std::cout<<c.C::GetData()<<std::endl;
因为C类中未重定义GetData(),故调用从B继承来的GetData(),但是B类也未定义,所以调用A中的GetData(),因为A中的doGetData()是虚函数,所以调用B类中的doGetData(),而B类的doGetData()返回B::m_data,故输出1。

std::cout<<c.doGetData()<<std::endl;
肯定是B类的返回值1了。

std::cout<<c.A::doGetData()<<std::endl;
因为直接调用A的doGetData(),返回输出0。

std::cout<<c.B::doGetData()<<std::endl;
因为直接调用了B的doGetData(),所以输出1。

std::cout<<c.C::doGetData()<<std::endl;
因为C类中未重定义该接口,所以调用B类中的doGetData(),而B类的doGetData()返回B::m_data,故输出1。这里要注意存在一个就近调用,如果父辈存在相关接口则优先调用父辈接口,如果父辈接口也不存在相关接口则调用祖父辈接口。

答案:






评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值