C++对象内存分布(1) - 私有虚函数

1.前言

本篇文章的所有代码例子,如果是windows上编译运行,则使用的是visual studio 2013。如果是RHEL6.5平台(linux kernal: 2.6.32-431.el6.i686)上编译运行,则其gcc版本为4.4.7,如下所示:

[[email protected] ~]# gcc --version

gcc (GCC) 4.4.7 20120313 (Red Hat 4.4.7-4)

2.非公有虚函数

在开始探讨对象中内存的具体分布之前,先来看下C++中虚函数的一些行为。

2.1.公有虚函数

子类通过覆盖公有虚函数的方式实现多态是最常见的情况。指向子类对象的基类指针调用被子类覆盖的函数,实际上,调用的是子类的函数。

参考以下代码:

#include <iostream>

class Base
{
public:
	virtual void foo()
	{
		std::cout << "Base::foo" << std::endl;
	}
};

class Derive :public Base
{
	void foo()
	{
		std::cout << "Derive::foo" << std::endl;
	}
};

int main()
{
	Base *pb = new Derive();
	pb->foo();
	return 0;
}

毫无疑问,虽然foo在子类中默认是private私有的,但输出结果会如下面所示:

Derive::foo

具体原因,在前面的C++系列文章中已经讨论过。具体可以参考这篇文章"C++虚函数(9) - 虚函数能否为private?"。

2.2.私有虚函数

和公有虚函数相对的是保护虚函数和私有虚函数,在这里,为了方便探讨问题,只讨论私有虚函数。至于保护虚函数的情况和私有虚函数大同小异,差异只在于函数的可见性不同。

从表面看,虚函数的作用就是为了使父类指针能够访问到子类对象的函数。如果将虚函数设置为私有的,那么,无论子类对象,还是父类对象,都无法访问到该函数。这样的函数就变得毫无意义了。实际情况果真如此吗?答案是否定的。首先C++支持私有虚函数,当然也支持保护虚函数,在这里,对保护虚函数不做讨论。

设计模式中有封装算法这一设计原则。该原则使用模板方法模式,向外部提供访问的接口,该接口明确了算法的步骤。对于算法的每个步骤,由外部通过继承的方式来指定。

下面举个简单的例子来说明。

众所周知,购物的过程是,先挑选商品,接着付款,最后离开商店。在这个过程中,3个步骤是有顺序的,不能随意地颠倒次序。在生活中,这个类似的例子很多。以下编写一个表示购物的抽象类,它拥有3个私有纯虚函数choose(),pay()和leave(),分别表示购物的挑选商品,付款和离开商店的3个过程,由shopping函数依次调用这3个函数:

#include <iostream>

class Store
{
public:
	void shopping()
	{
		std::cout << "Base::shopping" << std::endl;
		choose();
		pay();
		leave();
	}

private:
	virtual void choose() = 0;
	virtual void pay() = 0;
	virtual void leave() = 0;
};

//实现一个表示购买蔬菜的类,它继承自Store类,覆盖了Store类中的choose(),pay()和leave()函数。
class VegetableStore :public Store
{
private:
	void choose()
	{
		std::cout << "VegetableStore::choose" << std::endl;
	}

	void pay()
	{
		std::cout << "VegetableStore::pay" << std::endl;
	}

	void leave()
	{
		std::cout << "VegetableStore::leave" << std::endl;
	}
};

int main()
{
	Store *ps = new VegetableStore();
	ps->shopping();
	return 0;
}

输出:

Base::shopping

VegetableStore::choose

VegetableStore::pay

VegetableStore::leave

从上面的代码可以看出,父类Store通过shopping函数规定了购物的过程,至于购物的每个步骤该做什么,由子类来指定。由于表示购物步骤的函数被声明为私有的,我们就不能通过子类对象来调用choose(),pay()或lease(),避免了出现先离开商品,再挑选商品的混乱情况。

时间: 2025-01-02 14:42:18

C++对象内存分布(1) - 私有虚函数的相关文章

C++对象内存模型2 (虚函数,虚指针,虚函数表)

从例子入手,考察如下带有虚函数的类的对象内存模型: 1 class A { 2 public: 3 virtual void vfunc1(); 4 virtual void vfunc2(); 5 void func1(); 6 void func2(); 7 virtual ~A(); 8 private: 9 int m_data1, m_data2; 10 }; 11 12 class B : A { 13 public: 14 virtual void vfunc1();; 15 vo

类对象内存分布

================================================================================================ 如何计算类对象占用的字节数? 一个空类的话1个字节.这是为了保证n个连续空类声明时,变量地址有偏移,防止变量覆盖. 非空类的话用关键字sizeof计算.如果手工计算就相当麻烦,光padding就一堆规则了.而且有些额外信息比如虚函数(多个虚函数也只产生一个vptr指针)等等.一个类成员 ,当有虚函数时,有

C++对象内存分布(2) - 菱形继承(non virtual)

1.前言 本篇文章的所有代码例子,如果是windows上编译运行,则使用的是visual studio 2013.如果是RHEL6.5平台(linux kernal: 2.6.32-431.el6.i686)上编译运行,则其gcc版本为4.4.7,如下所示: [[email protected] ~]# gcc --version gcc (GCC) 4.4.7 20120313 (Red Hat 4.4.7-4) 2.菱形继承类的内存分布 2.1.类的结构 菱形继承 - 重复继承 2.2.实现

C++对象内存分布(3) - 菱形继承(virtual)

1.前言 本篇文章的所有代码例子,如果是windows上编译运行,则使用的是visual studio 2013.如果是RHEL6.5平台(linux kernal: 2.6.32-431.el6.i686)上编译运行,则其gcc版本为4.4.7,如下所示: [[email protected] ~]# gcc --version gcc (GCC) 4.4.7 20120313 (Red Hat 4.4.7-4) 2.菱形继承类的内存分布 本篇文章主要讨论的是虚继承(virtual)下的内存分

私有虚函数

#include<iostream> using namespace std; class base { private: virtual void func(){cout<<"base : func()"<<endl;} }; class derived :public base { public: virtual void func(){cout<<"derived :func()"<<endl;} }

c++派生类中构造函数和析构函数执行顺序、判断对象类型、抽象类、虚函数

一. 代码: 1 #include<stdio.h> 2 #include<string.h> 3 #include<algorithm> 4 #include<iostream> 5 using namespace std; 6 class A 7 { 8 public: 9 int a,b; 10 A(); 11 A(int x,int y); 12 ~A(); 13 }; 14 A::A() 15 { 16 printf("调用A类构造函数\

对象内存分布

对象在内存中分为三个区域:1 对象头(Header) 2 实例数据(Instance Date) 3 对齐填充(Padding) 1 对象头: 1) 存储自身运行时的数据,如Hascode ,GC 分代年龄,锁状态标记,线程持有锁,偏向线程id,偏向时间戳.数据存放在32bit 和64bit 机中分别用32bit和64bit来存储. 2)  对象头的另一部分是类型指针.是对象指向他的类元数据的指针.虚拟机通过这个指针来确认这个对象是哪个类的实例.注:不是所有的对象都有这个记录, 2 实例数据:

java的对象内存分布

1.变量是保存在栈里的 保存的是数据不是引用地址  所以一个变量是一个值, 一 一对应  int a = 1; int b = a; b = 2; //a结果为1,b结果为2  a并不会改变 2.对象的数据是保存在堆里面的 而引用地址是保存在栈里的  调用流程是先到栈获取引用地址,再通过引用地址再堆里找到对应的数据, 所以当某一个对象的值改变 其他引用该对象的值也会改变

[转载]多重继承及虚继承中对象内存的分布

粘过来的效果还不错:) 本位通过不断地完善讲解多重继承及虚继承中对象内存的分布. 读的时候不要着急,第一遍会有些晕,第二遍就会好很多. 能帮助加深对多重继承及虚继承对象内存的分布情况. 多重继承及虚继承中对象内存的分布     这篇文章主要讲解G++编译器中虚继承的对象内存分布问题,从中也引出了dynamic_cast和static_cast本质区别.虚函数表的格式等一些大部分C++程序员都似是而非的概念. 本文是介绍C++的技术文章,假定读者对于C++有比较深入的认识,同时也需要一些汇编知识.