irrlicht引擎源码剖析 - 引用计数_irrlicht ireferencecounted-程序员宅基地

技术标签: reference  引擎  多线程  IrrLicht  delete  class  graph  

在研究irrlicht的video driver和scene graph之前,必须先了解在irrlicht中广泛应用的引用计数机制。irrlicht的接口IReferenceCounted实现了引用计数的机制。需要采用引用计数管理的类都会继承这个接口。irr的引用计数最重要的两个接口就是grab()和drop()。

view plaincopy to clipboardprint?
bool drop() const 
{  
// someone is doing bad reference counting.  
            _IRR_DEBUG_BREAK_IF(ReferenceCounter <= 0)  
    --ReferenceCounter;  
    if (!ReferenceCounter)  
    {  
        delete this;  
        return true;  
    }  
    return false;  

bool drop() const
{
// someone is doing bad reference counting.
   _IRR_DEBUG_BREAK_IF(ReferenceCounter <= 0)
 --ReferenceCounter;
 if (!ReferenceCounter)
 {
  delete this;
  return true;
 }
 return false;
}

grab()只是简单的将引用计数加一,而drop()是这个机制的核心,当drop()后当引用计数为0时,对象实例通过delete this删除自己从而最终释放了无人再使用的对象。正确的使用grab()和drop()是引用计数机制的核心,当你想长期持有一个对象时,需要grab()他,然后在不再使用的时候drop()他。grab()后,这个对象就不会因为其他地方的drop而被释放掉,因为至少你还有对他的引用。如果只是在一个函数段里面短期的使用一个对象,可以不grab也不drop,当然你要考虑多线程的情况。另外如果对象是你创建的,但你不能决定什么时候释放他,你就应该再把该对象添加到某个对象后调用drop(),这种情况也很常用。比如下面的伪代码:

IImage* pImage = new CImage(...);

pImageManager->add(pImage); //in add, pImageManager will call a grab()

pImage->drop();

这样很容易将对象的创建和释放分开到不同的对象中。

当然引用计数机制也让很多人讨厌,因为用乱了用错了麻烦一大堆。如果是自己创建的对象还好,而irrlicht引擎中提供的对象就要小心对待了。一般说来使用getXXX()得到的对象不需要drop(),当然除非你grab()了。使用createXXX()或你自己new出来的对象,你需要保证他在某个地方被drop()了,实在不放心应该深入引擎的代码看一下,建议使用vld这样的可视化内存泄露监测工具帮助你查找泄露内存的地方。

另外还有两个值得注意的问题

1)IReferenceCounted接口往往是被虚拟继承的,例如

class ISceneManager : public virtual IReferenceCounted

因为使用引用计数的接口和类很多,有可能某个类实现了几个接口,而这几个接口都继承自IReferenceCounted, public virtual继承保证只有一个IReferenceCounted基类对象

例如class CSceneManager : public ISceneManager, public ISceneNode就是这种情况

因为:

class ISceneNode : virtual public io::IAttributeExchangingObject

class IAttributeExchangingObject : virtual public IReferenceCounted

而class ISceneManager : public virtual IReferenceCounted

2)观察一些irrlicht引擎中的接口,接口中没有定义虚拟的析构函数

例如class IMesh : public virtual IReferenceCounted,在include/IMesh.h中

这个接口里面连~IMesh()都没有,更不会有virtual ~IMesh()了。

按照c++通常的思维,如果一个类明确的是要被继承的,需要定义一个virtual析构。那么为什么irrlicht中的很多接口都不定义呢?正因为他是引用计数的。比如,有一个接口IBase和他的子类CDerived。如果这么用

IBase* pBase = new CDerived();

那么当delete pBase时,如果IBase没有虚拟析构,就不会正确执行CDerived的析构了。

但是如果IBase继承自IReferenceCounted,通常我们的用法不会使用delete pBase来释放这个对象了。我们会使用pBase->drop(),问题就在这儿,这个drop()就是IReferenceCounted接口定义的那个,最终会执行delete this,这个this的类型就是调用drop()的对象的类型,这个对象实际是子类的,所以会调用子类的析构函数,从而会进一步调用父类的析构,所以父类没有定义虚拟析构也不要紧了。这就是引用计数的花招。当然irrlicht引擎中有些接口也定义了虚拟析构,这没有影响,这样更安全,即使某人没有用引用计数的方法管理对象的生命周期也不会出问题。我觉得,无论是否使用这样的引用计数机制,最好给父类都加上虚拟析构比较安全,也比较明确。

 

本文来自程序员宅基地,转载请标明出处:http://blog.csdn.net/n5/archive/2009/07/12/4342758.aspx

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/pizi0475/article/details/5633362

智能推荐

lufylegend.js的简单使用-程序员宅基地

文章浏览阅读826次。js 引入<script type="text/javascript" src="js/lufylegend/lufylegend-1.10.1.min.js"></script>html<div id="legend"></div><img src="" alt="" class="photo_img">..._lufylegend.js

MongoDB入门级保姆教程_mongodb保姆间教程-程序员宅基地

文章浏览阅读687次,点赞3次,收藏4次。MongoDB是文档数据库,旨在简化开发和扩展,本文主要介绍关键概念和基础语句并提供操作和管理上的注意事项。_mongodb保姆间教程

输入若干个正整数,判断每个数从高位到低位各位数字是否按值从小到大排列_输入一批正整数(以零或负数为结束标志),判断每个数从高位到低位的各位数字是否按-程序员宅基地

文章浏览阅读1.1w次,点赞7次,收藏16次。4-2输入若干个正整数,判断每个数从高位到低位各位数字是否按值从小到大排列,请根据题意,将程序补充完整。#include <stdio.h>int fun1(int m);int main(void){ int n; scanf("%d", &n); while (n > 0) { if(fun1(..._输入一批正整数(以零或负数为结束标志),判断每个数从高位到低位的各位数字是否按

IDEA用户登入简易版_idea中的个人博客网址的登录网址是那个去那里找-程序员宅基地

文章浏览阅读281次。<!DOCTYPE html><html lang="en"><head> <style> form{ border: 10px solid cornflowerblue;border-radius: 10px } </style> <meta charset="UTF-8"> <title>用户注册</title>..._idea中的个人博客网址的登录网址是那个去那里找

使用 Python 数据写入 Excel 工作表_python把数据写表格-程序员宅基地

文章浏览阅读2.4k次,点赞40次,收藏39次。本文所使用的 API 中,使用 Workbook 类来代表一个 Excel 工作簿。在操作 Excel 工作簿时,可以使用该类下的 LoadFromFile() 方法从文件读取 Excel 工作簿进行操作或直接通过创建 Workbook 的对象从而创建工作簿进行操作。需要注意的是,新建的 Excel 工作簿默认有三个工作表。同时,该 API 还提供 Worksheet 类和一系列方法、属性来对工作表及其中的单元格数据、格式等内容进行操作。_python把数据写表格

关于No converter found for return value of type: class java.util.ArrayList出现的几个问题-程序员宅基地

文章浏览阅读6.4k次,点赞5次,收藏4次。当我使用spring,springmvc,mybatis整合开发项目的时候,在controller层的方法使用@responsebody想要返回一个list集合对象的转换为json格式在页面输出。出现了异常:No converter found for return value of type: class java.util.ArrayList,是说明没有可以转换对象成json的转换..._no converter found for return value of type: c

随便推点

浙大 | PTA 习题7-7 字符串替换 (15分)_例题3-7 统计英文字母和数字字符[2] 分数 15 作者 颜晖 单位 浙大城市学院 本题要-程序员宅基地

文章浏览阅读2.3k次。本题要求编写程序,将给定字符串中的大写英文字母按以下对应规则替换:原字母 对应字母 A Z B Y C X D W … … X C Y B Z A输入格式:输入在一行中给出一个不超过80个字符、并以回车结束的字符串。输出格式:输出在一行中给出替换完成后的字符串。输入样例:Only the 11 CAPItaL LeTtERS are replaced.输出样例:..._例题3-7 统计英文字母和数字字符[2] 分数 15 作者 颜晖 单位 浙大城市学院 本题要

Bioinformatics | 预测药物-药物相互作用的多模态深度学习框架_ddimdl-程序员宅基地

文章浏览阅读4.1k次,点赞4次,收藏38次。作者 | 朱玉磊审稿 | 李芬今天给大家介绍来自华中农业大学信息学院章文教授课题组在Bioinformatics上发表的一篇关于预测药物与药物相互作用事件的文章。作者提出了一个多模态深度..._ddimdl

制作一个有趣的QQ机器人_qrspeed官网-程序员宅基地

文章浏览阅读7.7k次,点赞19次,收藏72次。如何制作一个有趣的QQ机器人制作一个好玩的QQ机器人(只能手机进行操作哦)题记:这个机器人用来整蛊兄弟或者是在朋友面前装逼都是不错的选择QQ机器人简介机器人效果图机器人制作方法机器人必下软件如何制作机器人词库的编写编写词库的软件词库的编写规则给大家找了一个QR下载的官网(不想加群的兄弟姐妹看这个)结尾题记:这个机器人用来整蛊兄弟或者是在朋友面前装逼都是不错的选择)QQ机器人简介QQ机器人,根据字面意思,就是利用特定的代码,使一个QQ账号成功具备自我反应并作出应答,而这也是我今天想要教你们做的一款最_qrspeed官网

「离散数学」是一门什么样的学科_离散数学学什么-程序员宅基地

文章浏览阅读2.4k次,点赞6次,收藏13次。写这篇文章的动机是想探讨从离散数学开始入门数理逻辑的路径以及离散数学与数理逻辑之间的关系。以学习数理逻辑为目的学习离散数学,而一般的以学习计算机为目的的学习还是有相当的不同,最大的不同就是:以数理逻辑为目的的学习,应当以「证明」 — — 形式证明为目的,这其中包括了关于形式证明的理论 — — 一阶理论的句法和语义,以及关于形式证明的实践 — — 证明框架和策略。学习的中心内容有两个:「语言」 — — 「 一阶语言」;「结构」 — — 数学中关于「结构」的思想、概念、种类、实例以及「结构」和「语言」的关系。_离散数学学什么

使用 vue-cli 遇到的坑1 - 打包后显示空白_vue cli 组件导出为null-程序员宅基地

文章浏览阅读248次。输入 npm run build 打包后显示空白解决方法1. 打开 config 文件夹里的 index.js 文件。找到 build 下面的 assetsPublicPath,将原来的assetsPublicPath: '/'修改为assetsPublicPath: './'2.打开 build 文件夹里的 utils.js,在下图位置添加 publicPath: '../../'..._vue cli 组件导出为null

嵌入式Linux(二十二)Linux内核分析及移植_嵌入式内核移植qspi-程序员宅基地

嵌入式Linux内核分析及移植的文章介绍了编译Linux内核的过程,并提供了一个编译脚本。作者使用NXP提供的内核进行移植,并在自己的开发板上进行测试。在启动Linux后,可以使用ifconfig命令查看网络接口,并使用ifconfig eth0 up命令启动ENET2接口。

推荐文章

热门文章

相关标签