怎样在一个容器中包含类型不同,但是彼此有关系的对象?众所周知,C++的容器只能存放类型相同的元素,所以直接在一个容器中存储不同类型的对象本身是不可能的,只能通过以下两种方案实现:

1. 提供一个间接层,在容器中存放对象的指针而不是对象本身。

2. 通过代理类实现。

class Animal
{
public:
virtual void Eat() = ;
}; class Cat : public Animal{};
class Dog : public Animal{};
class Bird : public Animal{};

在上面我们看到有一个虚基类和三个继承类,下面分别用两种方案来实现一个容器存放不同类型但又互相关联的类。

1.通过指针实现

Animal* animals_array[];
Cat cat;
Dog dog;
Bird bird;
animals_array[] = &cat;
animals_array[] = &dog;
animals_array[] = &bird;

这样会带来一个问题,就是容器中的指针指向的对象如果被销毁,这个指针就会变成野指针,就像下面这样:

Animal* animals_array[];
do
{
Cat cat;
Dog dog;
Bird bird;
animals_array[] = &cat;
animals_array[] = &dog;
animals_array[] = &bird;
}while();
//此时对象已经被析构,容器中的指针指向未知内容

也可以换一种方式,构造一个新的动态对象,将其地址放在容器中,这样就可以避免对象析构导致指针失效的问题:

Cat cat;
animals_array[] = new Cat(cat);

这样会曾加额外的内存开销,并且可能出现容器中两个指针同时指向一个对象的情况。
所以,在容器中存放不同对象的指针并不是一个很好的解决方案。

2.通过代理类实现

实现代码如下:

class Animal
{
public:
virtual void Eat() = ; //copy函数,构造一个基于自身对象类型的对象
virtual Animal* copy() const = ; virtual ~Animal() {}
}; class Cat : public Animal
{
public:
virtual void Eat()
{
std::cout << "cat eat." << std::endl;
} virtual Animal* copy() const
{
// 返回一个以自身作为参数构造的Cat类型的对象
return new Cat(*this);
}
}; class Dog : public Animal
{
public: virtual void Eat()
{
std::cout << "dog eat." << std::endl;
} virtual Animal* copy() const
{
// 返回一个以自身作为参数构造的Dog类型的对象
return new Dog(*this);
}
}; class Bird : public Animal
{
public: virtual void Eat()
{
std::cout << "bird eat." << std::endl;
} virtual Animal* copy() const
{
// 返回一个以自身作为参数构造的Bird类型的对象
return new Bird(*this);
}
}; //代理类
class AnimalSurrogate
{
public:
AnimalSurrogate() :pa(NULL) {} AnimalSurrogate(const Animal& ani)
{
pa = ani.copy();
} //拷贝构造
AnimalSurrogate(const AnimalSurrogate& ani_srg)
{
pa = ani_srg.pa != nullptr ? ani_srg.pa->copy() : nullptr;
} ~AnimalSurrogate()
{
if (pa != nullptr)
{
delete pa;
pa = nullptr;
}
} //重载 = 操作符
AnimalSurrogate& operator=(const AnimalSurrogate& ani_srg)
{
if (this != &ani_srg)
{
delete pa;
pa = ani_srg.pa != nullptr ? ani_srg.pa->copy() : nullptr;
}
return *this;
} //将基类中的公共函数搬过来,这样就可以通过代理类直接访问这些方法
void Eat()
{
if (pa == nullptr)
{
throw "empty AnimalSurrogate.Eat()";
}
return pa->Eat();
} private:
Animal* pa;//存储基类的指针
};

通过代码可以看出来,所谓的代理类,就是构造一个新的类,这个类中包含关联的基类类型的指针,该指针可以指向不同类型但又相互关联的子类对象,通过指针可以转调对象的方法,同时实现内存的管理。代理类的实用方法如下:

Cat cat;
Dog dog;
Bird bird; arr[] = AnimalSurrogate(cat);
arr[] = AnimalSurrogate(dog);
arr[] = AnimalSurrogate(bird); arr[].Eat();//输出 cat eat.
arr[].Eat();//输出 dog eat.
arr[].Eat();//输出 bird eat.

总结:代理类的的每个对象都代表另一个对象,该对象可以使位于一个完成继承层次中的任何类的对象。通过在容器中用代理对象而不是对象本身的方式,实现容器中存放不同类型的对象

使用代理类的优缺点如下:

  • 优点:使用代理类比直接在容器中存放不同对象的指针更安全,便于实现内存管理。
  • 缺点:内存开销太大,放入容器的每个对象都需要拷贝一次,不够灵活。

为了避免对象的拷贝,可以通过句柄类来实现,关于句柄类的原理和使用在下一篇问文章中作介绍。

C++的代理类的更多相关文章

  1. vs2013 手动生成webservice代理类wsdl

    第一步: 第二步: 第三步: 至此wsdl代理类生成成功!

  2. Struts2 源码分析——Action代理类的工作

    章节简言 上一章笔者讲到关于如何加载配置文件里面的package元素节点信息.相信读者到这里心里面对struts2在启动的时候加载相关的信息有了一定的了解和认识.而本章将讲到关于struts2启动成功 ...

  3. iOS Class 使用NSProxy和NSObject设计代理类的差异

    经常发现在一些需要使用消息转发而创建代理类时, 不同的程序员都有着不同的使用方法, 有些采用继承于NSObject, 而有一些采用继承自NSProxy. 二者都是Foundation框架中的基类, 并 ...

  4. WCF通过SVCUtil.exe生成客户端代理类和配置文件(转)

    WCF服务调用通过两种常用的方式: 1:一种是借助代码生成工具SvcUtil.exe或者添加服务引用的方式. 2:一种是通过ChannelFactory直接创建服务代理对象进行服务调用. 本文只针对通 ...

  5. 终于解决:升级至.NET 4.6.1后VS2015生成WCF客户端代理类的问题

    在Visual Studio 2015中将一个包含WCF引用的项目的targetFramework从4.5改为4.6.1的时候,VS2015会重新生成WCF客户端代理类.如果WCF引用配置中选中了&q ...

  6. Web Service代理类生成工具

    本文原文连接:http://www.cnblogs.com/dengxinglin/p/3334158.html 之前一篇文章写 Web Service服务代理类生成及编译 , 通过命令行的方式可以直 ...

  7. Web Serveice服务代理类生成及编译

    本文链接地址:http://www.cnblogs.com/dengxinglin/p/3334158.html 一.生成代理类 对于web service服务和wcf的webservice服务,我们 ...

  8. 解析利用wsdl.exe生成webservice代理类的详解

    利用wsdl.exe生成webservice代理类:根据提供的wsdl生成webservice代理类1.开始->程序->Visual Studio 2005 命令提示2.输入如下红色标记部 ...

  9. 【C++沉思录】代理类

    1.考虑下面的场景:设计一个容器,包含一组类型不同但相互关联的对象(比如:Animal,Dog,Cat),对象具备多态行为.2.容器一般只能包含一种类型的对象,使用vector<Animal&g ...

  10. EF容器---代理类对象

    #region 修改--官方的修改是,先查询,然后修改 /// <summary> /// 修改--官方的修改是,先查询,然后修改 /// </summary> static ...

随机推荐

  1. 使用mpvue开发微信小程序

    更多内容请查看 我的新博客 地址 : 前言 16年小程序刚出来的时候,就准备花点时间去学学.无奈现实中手上项目太多,一个接着一个,而且也没有开发小程序的需求,所以就一拖再拖. 直到上周,终于有一个小程 ...

  2. MognoDB3.4.2用户访问配置管理

    说在前面,官方文档似乎略有瑕疵. 说一下大规则:把绑定IP换成127.0.0.1IP之后,把security的authorization关闭掉做用户添加操作. 添加用户的方法必须是createUser ...

  3. pydev 下Django 1.7 undefined variables from import问题的解决

    参考:http://stackoverflow.com/questions/24951029/pydev-django-undefined-variables-from-import 参考上面的帖子认 ...

  4. ubuntu 14.04 安装svn server (subversionedge )

    ubuntu 14.04 安装subversionedge 请仔细阅读安装包自带的readme文件! 1.先去官网,找安装包: http://subversion.apache.org/ http:/ ...

  5. mac上php版本切换

    目标:Mac 环境下完成 php 版本之间的切换 在本地开发中很多时候我们需要多个版本的 php 开发环境.在公司中习惯用自己电脑开发的伙伴们,常常因为公司线上环境被迫更换php版本.但有不想降低自己 ...

  6. 用js来实现那些数据结构16(图02-图的遍历)

    上一篇文章我们简单介绍了一下什么是图,以及用JS来实现一个可以添加顶点和边的图.按照惯例,任何数据结构都不可或缺的一个point就是遍历.也就是获取到数据结构中的所有元素.那么图当然也不例外.这篇文章 ...

  7. IT连创业系列:近期功能调整(小魔术功能从二级目录调整到一级栏目)

    最近花了点时间,折腾了一下.NET Core,因此有几篇 Taurus.MVC + CYQ.Data 的文章出来. 这两天也顺带把 ASP.NET Aries 升级了一下功能, 也计划支持.NET C ...

  8. Effective Java 第三版——42.lambda表达式优于匿名类

    Tips <Effective Java, Third Edition>一书英文版已经出版,这本书的第二版想必很多人都读过,号称Java四大名著之一,不过第二版2009年出版,到现在已经将 ...

  9. 页面读取Excel

    var input = document.getElementById("file");    //支持chrome IE10    if (window.FileReader) ...

  10. FutureTask理解

    一.概述 FutureTask包装器是一种非常便利的机制,同时实现了Future和Runnable接口. 类图如下: FutureTask是一种可以取消的异步的计算任务.它的计算是通过Callable ...