句柄类存在的意义是为了弥补将派生类对象赋给基类对象时发生的切片效应。比如以下的程序:

multimap<Base> basket;
Base base;
Derived derive;
basket.insert(base); //ok,add copy of base;
basket.insert(derive); //ok,but derive sliced down to its base part.

也就是说在把派生类的对象赋值给基类的时候,会发生切片效益,派生类的非基类部分会被切掉,那么就失去了本身的意义。为了解决问题我们能够

使用基于基类的指针或者引用,可是设计到指针问题的话就涉及到资源不使用后的释放问题。这就引出了句柄类,它类似智能指针,能够在我们复制资源

的时候不用去操心内存泄露的问题。整个程序的设计例如以下所看到的:

//Base.h
#pragma once
class Base
{
public:
Base(void);
virtual ~Base(void);
virtual Base* clone() const;
virtual void fine() const;
private:
int mb;
};
//Base.cpp
#include "Base.h"
#include <iostream>
using namespace std; Base::Base(void):mb(12)
{
} Base::~Base(void)
{
} Base* Base::clone() const
{
return new Base(*this);
} void Base::fine() const
{
cout<<"Base fine function"<<endl;
}
//Derive.h
#pragma once
#include "base.h"
class Derive :
public Base
{
public:
Derive(void);
virtual ~Derive(void);
virtual Derive* clone() const;
virtual void fine() const;
private:
int md;
};
//Derive.cpp
#include "Derive.h"
#include <iostream>
using namespace std; Derive::Derive(void):Base(),md(13)
{
} Derive::~Derive(void)
{ } Derive* Derive::clone() const
{
return new Derive(*this);
} void Derive::fine() const
{
cout<<"Derive fine function"<<endl;
}
//Handles.h
#pragma once
#include "Base.h"
#include <iostream>
class Handles
{
public:
Handles(void);
Handles(const Base&);
Handles(const Handles& h);
~Handles(void); const Base* operator->()const;
const Base& operator*()const;
Handles& operator=(const Handles&); private:
Base* p;
std::size_t* use;
void dec_use()
{
if(--*use == 0)
{
delete p;
delete use;
std::cout<<"delete resource"<<std::endl;
}
} };
//Handle.cpp
#include "Handles.h" Handles::Handles(void):p(NULL),use(new size_t(1))
{
} Handles::Handles(const Handles& h):p(h.p),use(h.use)
{
++*use;
} Handles::Handles(const Base& item):p(item.clone()),use(new std::size_t(1))
{
}
const Base& Handles::operator*()const
{
if(p)
return *p;
else
throw std::logic_error("unbounded Handles");
} const Base* Handles::operator->()const
{
if(p)
return p;
else
throw std::logic_error("unbounded Handles");
} Handles& Handles::operator=(const Handles& h)
{
++*h.use;
dec_use();
p = h.p;
use = h.use;
return *this;
} Handles::~Handles()
{
dec_use();
}
//main.cpp
#include <iostream>
#include "Handles.h"
#include "Derive.h"
#include <vector>
using namespace std;
void main()
{
vector<Handles> mb;
Base b;
Derive d;
mb.push_back(Handles(b));
mb.push_back(Handles(d));
mb[0]->fine();
mb[1]->fine();
system("pause");
}

C++ 句柄类的原理以及设计的更多相关文章

  1. C++中的句柄类

    初次在<C++ Primer>看到句柄,不是特别理解.在搜索相关资料后,终于有了点头绪. 首先明白句柄要解决什么问题.参考文章<C++ 沉思录>阅读笔记——代理类 场景: 我们 ...

  2. c++ 容器、继承层次、句柄类

    一.容器与继承 在容器中保存有继承关系的对象,如果定义成保存基类对象,则派生类将被切割,如果定义成保存派生类对象,则保存基类对象又成问题(基类对象将被强制转换成派生类对象,而派生类中定义的成员未被初始 ...

  3. C++ 句柄类

    一.容器与继承 在容器中保存有继承关系的对象时,如果定义成保存基类对象,则派生类将被切割,如果定义成保存派生类对象,则保存基类对象又成问题(基类对象将被强制转换成派生类对象,而派生类中定义的成员未被初 ...

  4. [三]java8 函数式编程Stream 概念深入理解 Stream 运行原理 Stream设计思路

    Stream的概念定义   官方文档是永远的圣经~     表格内容来自https://docs.oracle.com/javase/8/docs/api/   Package java.util.s ...

  5. 2017-2018-1 20179205《Linux内核原理与设计》第六周作业

    <Linux内核原理与设计> 视频学习及操作 给MenuOS增加time和time-asm命令的方法: 1.更新menu代码到最新版 rm menu -rf //强制删除menu, rm ...

  6. atitit.  web组件化原理与设计

    atitit.  web组件化原理与设计 1. Web Components提供了一种组件化的推荐方式,具体来说,就是:1 2. 组件化的本质目的并不一定是要为了可复用,而是提升可维护性. 不具有复用 ...

  7. Atitit.ati dwr的原理and设计 attilax 总结 java php 版本

    Atitit.ati dwr的原理and设计 attilax 总结 java php 版本 1. dwr的优点相对于ajax来说..1 2. DWR工作原理1 3. Dwr的架构2 4. 自定义dwr ...

  8. C++的句柄类

    上一篇文件介绍了关于C++代理类的使用场景和实现方法,但是代理类存在一定的缺陷,就是每个代理类会创建一个新的对象,无法避免一些不必要的内存拷贝,本篇文章引入句柄类,在保持代理类多态性的同时,还可以避免 ...

  9. Atitit.ati&#160;dwr的原理and设计&#160;attilax&#160;总结&#160;java&#160;php&#160;版本号

    Atitit.ati dwr的原理and设计 attilax 总结 java php 版本号 1. dwr的长处相对于ajax来说.. 1 2. DWR工作原理 1 3. Dwr的架构 2 4. 自己 ...

随机推荐

  1. Sqoop 产生背景(一)

    Sqoop 的产生主要源于: 1.目前很多使用hadoop技术的企业,有大量的数据存储在传统关系型数据库中. 2.早期由于工具的缺乏,hadoop与传统数据库之间的数据传输非常困难. 1)传统数据库中 ...

  2. 使用A*寻路小记

    前几天做另一个DEMO 要用实现自动寻路功能,看到普遍都是A* 学习了下 我的主循环代码: isFindEndPoint = false; //主循环 do { CreateOutSkirtsNode ...

  3. CF615C Running Track

    思路: kmp + 二分. 实现: #include <iostream> #include <cstdio> #include <algorithm> #incl ...

  4. Floyd模板

    比较简单的算法:但是当点太多需要剪枝,不然很耗时 void Floyd() { ;k<n;++k) ;i<n;++i) ;j<n;++j) dj[i][j] = min(dj[i][ ...

  5. 如何用java生成随机验证码

     1.VerifyCode 类:   1 package com.HRuinger.enity;                          ImageIO.write(image, " ...

  6. 3星|《刷新》:微软第三任CEO上任三年后的回顾

    刷新:重新发现商业与未来 作者是微软第三任CEO,2014年2月上任.本书英文版出版于2017年9月,全书内容大致截至于2017年年初,算是作者上任三年后的回顾. 书中作者讲了不少自己的个人经历.作者 ...

  7. SpringMVC注解配置处理器映射器和处理器适配器

    一.springmvc.xml中配置方式 <!--注解映射器 --> <bean class="org.springframework.web.servlet.mvc.me ...

  8. 【转】jmeter定时调度,持续并发,使用简介

    一.安装Jmeter 1.下载Jmeter 下载地址:http://jmeter.apache.org/download_jmeter.cgi 目前最新版为2.9,其余文件如源代码等也可从如下官网下载 ...

  9. c/c++排坑(4) -- c/c++中返回局部变量

    返回c语言中的局部变量 先看一段代码猜猜,打印值: #include <iostream> using namespace std; char * func(); int main() { ...

  10. enote笔记语言(3)

    what&why(why not)&how&when&where&which:紫色,象征着神秘而又潜蕴着强大的力量,故取紫色. key&key-memo ...