在成员函数中调用虚函数例子

#include<iostream>
#include<string>
using namespace std;

///考察的是类作用域、覆盖和隐藏,参考primer c++的p547 
//在成员函数中调用虚函数 https://blog.csdn.net/li_haoren/article/details/86165690
class one{
public:
	one(int data = 0):mdata(data){
		cout<<"one"<<endl;
	}
//	one(){
//		cout<<"one"<<endl;
//	}
	int data(){
		return get_data();  //这条语句是多态的 
	//“get_data();”等价于“this->get_data();”,而this指针显然是one*类型的,
	//即是一个基类指针,那么“this->func2();”就是在通过基类指针调用虚函数,因此这条函数调用语句就是多态的
	}
	virtual int get_data(){
		return mdata;
	}
protected:
	int mdata;
};

class two:public one{
public:
	two(int data = 1):mdata(data){
		cout<<"two"<<endl;
	}
	int get_data(){
		return mdata;
	}
protected:
	int mdata;
};

class three:public two{
public:
	three(int data = 2):mdata(data){
		cout<<"three"<<endl;
	}
//	int get_data(){         //加上输出就是:3 3 3 3                                                      
//		return mdata;                    // 3 0 1 3  
//	}
protected:
	int mdata;
};

int main(){
	three t(3);                                                                  //其实继承对成员函数的本质是:可以访问父类作用域的函数 
	cout<<t.data()<<" "<<t.one::data()<<" "<<t.two::data()<<" "<<t.three::data()<<" "<<endl  
	///本来就是调用的就是继承下来的one::的data(),动态绑定时传入的this是第三个类的,所以都是调用的第三个类的,而第三类没有重写虚函数,
	///用的是从第二个类(重写了虚函数)中继承下来的,所以真正调用的是第二个类的 

//下面的输出不会出现动态绑定,因为没有通过指针或者引用调用虚函数。
//考察的是类作用域,three类没有重新定义get_data()函数,所以会从其父类中找,找到了two中的get_data()函数,所以t.get_data()执行的是two中的函数,输出的也是two中的成员,值为1。
	 <<t.get_data()<<" "<<t.one::get_data()<<" "<<t.two::get_data()<<" "<<t.three::get_data()<<endl; 
	 //最后输出:
	 //1 1 1 1
     //1 0 1 1
}

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C++,构造函数调用虚函数是一个有趣的问题。根据引用\[1\]的解释,当调用构造函数时,对象的虚表指针已经被设定好,因此调用虚函数不会有任何问题。在引用\[2\]的示例代码,当创建一个B对象时,B的构造函数会先调用A的构造函数,然后调用B的构造函数。在A的构造函数调用虚函数fun(),此时对象的虚表指针已经指向B的虚表,因此调用的是B的fun()函数。同样,在B的构造函数调用虚函数fun()时,也会调用B的fun()函数。 然而,需要注意的是,在构造函数调用虚函数时,可能会导致意外的行为。因为在派生类的构造函数基类的成员变量还没有被初始化,因此在调用虚函数时,可能会访问到未定义的成员变量。这在引用\[3\]的示例代码可以看到。在这个例子,当创建一个Car对象时,Car的构造函数会先调用Base的构造函数,然后初始化自己的成员变量。在Base的构造函数调用虚函数Test(),此时Car对象的成员变量m_nValue还没有被初始化,因此在调用Test()时会导致未定义的行为。 因此,尽量避免在构造函数调用虚函数,以免引起意外的行为。如果需要在构造函数执行某些操作,可以考虑使用非虚函数或将虚函数延迟到对象完全构造完成后再调用。 #### 引用[.reference_title] - *1* *2* [在构造函数调用虚函数](https://blog.csdn.net/songchuwang1868/article/details/96481853)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [构造函数调用虚函数](https://blog.csdn.net/Think88666/article/details/104512651)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值