青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

清風(fēng)竹林

ぷ雪飄絳梅映殘紅
   ぷ花舞霜飛映蒼松
     ----- Do more,suffer less

基類(lèi)角色之對(duì)象管理器

基類(lèi)角色之對(duì)象管理器

版本:0.1

最后修改:2009-07-02

撰寫(xiě):李現(xiàn)民


問(wèn)題描述

C++程序設(shè)計(jì)中,保存一個(gè)生命周期不是由類(lèi)對(duì)象自己維護(hù)的其它對(duì)象的指針通常是個(gè)壞主意,因?yàn)槌绦蜻壿嫼茈y判斷在使用該指針的時(shí)刻其所指對(duì)象是否已經(jīng)被銷(xiāo)毀。這種應(yīng)用需求很常見(jiàn),例如在網(wǎng)游設(shè)計(jì)中,由于華麗的裝備加載需要進(jìn)行大量硬盤(pán)I/O,因此加載過(guò)程通常由一個(gè)獨(dú)立的加載線程執(zhí)行,由于在裝備加載完成的時(shí)刻該玩家很可能已經(jīng)下線,因此加載線程就需要能夠去判斷此時(shí)玩家對(duì)象是否仍然有效。

為了解決該問(wèn)題,通常會(huì)設(shè)計(jì)一個(gè)PlayerManager類(lèi)用于跟蹤管理當(dāng)前所有的玩家對(duì)象,而加載線程通過(guò)提供玩家id以確認(rèn)該玩家對(duì)象仍然存在。此種設(shè)計(jì)方案需要一個(gè)獨(dú)立的PlayerManager類(lèi),并提供一個(gè)全局的PlayerManager類(lèi)對(duì)象以跟蹤當(dāng)前的所有玩家對(duì)象。

出于代碼復(fù)用的目的,我希望實(shí)現(xiàn)一個(gè)通用基類(lèi)解決此類(lèi)問(wèn)題。該基類(lèi)需要為子類(lèi)對(duì)象至少提供以下幾方面的能力:

  1. 為所有的對(duì)象分配一個(gè)全局唯一的index,通過(guò)該index能夠(盡可能快的)獲取到擁有該index的類(lèi)對(duì)象(或NULL);

  2. 自動(dòng)跟蹤類(lèi)對(duì)象的生成與銷(xiāo)毀,不需要手工編寫(xiě)額外代碼;

  3. 實(shí)現(xiàn)迭代器,提供遍歷當(dāng)前所有有效對(duì)象的能力;

  4. 提供“移除”接口,使得對(duì)象可以主動(dòng)要求放棄被對(duì)象管理器跟蹤;

  5. 各子類(lèi)實(shí)現(xiàn)擁有完全獨(dú)立的管理器邏輯;


解決方案

將實(shí)現(xiàn)代碼保存為objectman.hpp,內(nèi)容如下:

/********************************************************************
created:    2009-06-29
author:    lixianmin

purpose:    base class for object manager
Copyright (C) 2009 - All Rights Reserved
********************************************************************
*/
#ifndef _LIB_OBJECT_MAN_HPP_INCLUDED_
#define _LIB_OBJECT_MAN_HPP_INCLUDED_
#include 
<cassert>
#include 
<map>

namespace lib
{
    template
<typename T>
    
class objectman
    {
    
public:
        typedef 
int index_t;                                // 索引類(lèi)型
        typedef std::map<index_t, T*>   object_map;         // 容器類(lèi)型

        
enum { INVAID_INDEX= 0};                            // 無(wú)效索引

    
public:
        
// 迭代器
        class iterator
        {
        
public:
            iterator(
void): _iter(_mObjects.begin()){}                          // 構(gòu)造函數(shù)
            bool has_next(voidconst { return (_mObjects.end()!= _iter); }     // 測(cè)試是否還有下一個(gè)對(duì)象
            T* next(void) { return has_next()? (_iter++->second): NULL;    }    // 獲取下一個(gè)對(duì)象指針

        
private:
            typename object_map::iterator _iter;
        };

    
public:
        
// 構(gòu)造函數(shù)
        objectman(void)
        {
            enable_index();
        }

        
// copy 構(gòu)造函數(shù)
        objectman(const objectman& rhs)
        {
            enable_index();
        }

        
// 析構(gòu)函數(shù)
        virtual ~objectman(void)
        {
            disable_index();
        }

        
// 賦值操作符
        objectman& operator= (const objectman& rhs)
        {

        }

        
// 通過(guò)索引獲取對(duì)象
        static T* get_by_index(index_t index)
        {
            object_map::iterator iter
= _mObjects.find(index);
            
if (_mObjects.end()!= iter)
            {
                T
* pObject= iter->second;
                assert(NULL
!= pObject);
                
return pObject;
            }

            
return NULL;
        }

        
// 獲取對(duì)象索引
        index_t get_index(voidconst { return _idxObject; }

        
// 生成索引(使能被對(duì)象管理器遍歷到)
        void enable_index(void)
        {
            _idxObject
= ++_idxGenderator;
            assert(get_index()
!= INVAID_INDEX);
            assert(_mObjects.find(get_index())
== _mObjects.end());
            _mObjects.insert(std::make_pair(get_index(), static_cast
<T*>(this)));
        }
        
        
// 移除索引(使不能被對(duì)象管理器遍歷到)
        void disable_index(void)
        {
            
if (get_index()!= INVAID_INDEX)
            {
                assert(_mObjects.find(get_index())
!= _mObjects.end());
                _mObjects.erase(get_index());

                _idxObject
= INVAID_INDEX;
            }
        }

    
private:
        friend 
class iterator;
        
static object_map    _mObjects;                          // 對(duì)象容器
        static index_t        _idxGenderator;                    // 索引發(fā)生器

        index_t                _idxObject;                      
// 對(duì)象索引
    };

    template
<typename T>
    typename objectman
<T>::object_map objectman<T>::_mObjects;

    template
<typename T>
    typename objectman
<T>::index_t objectman<T>::_idxGenderator= 0;
}

#endif

測(cè)試代碼

測(cè)試代碼如下:

#include <cassert>
#include 
"objectman.hpp"
// 聲明一個(gè)類(lèi)
class Player:public lib::objectman<Player>
{

};

int main(int argc, char* argv[])
{
    
const int idxDisabled= 5;
    
// 生成對(duì)象
    for (int i= 0; i< 10++i)
    {
        Player
* pPlayer= new Player;
        
if (idxDisabled== pPlayer->get_index())
        {
            
// 從對(duì)象管理器中移除該對(duì)象
            pPlayer->disable_index();
        }
    }
    
    
//使用迭代器遍歷類(lèi)對(duì)象
    Player::iterator iter;
    
while(iter.has_next())
    {
        Player
* pPlayer= iter.next();
        
const int idxPlayer= pPlayer->get_index();
        
// 斷言之:遍歷不到已經(jīng)移除的對(duì)象
        assert(idxPlayer!= idxDisabled);
        
// 斷言之:可以通過(guò)idxPerson取得對(duì)象指針
        assert(pPlayer== Player::get_by_index(idxPlayer));
        
        
// 回收對(duì)象
        delete pPlayer;
        pPlayer
= NULL;
    }

    
// 斷言之:所有對(duì)象均已被刪除
    Player::iterator iter2;
    assert(
!iter2.has_next());

    system(
"pause");
    
return 0
}

vs2008下所有斷言均動(dòng)作通過(guò)。


已知問(wèn)題

  1. 在大量生成對(duì)象的情況下,index索引空間(代碼定義為int的范圍)有可能使用殆盡,甚至產(chǎn)生重復(fù),這會(huì)導(dǎo)致兩個(gè)對(duì)象擁有相同index的嚴(yán)重錯(cuò)誤;

  2. std::map的查找速度不是特別另人滿(mǎn)意;






posted on 2009-07-02 12:49 李現(xiàn)民 閱讀(2049) 評(píng)論(11)  編輯 收藏 引用 所屬分類(lèi): design

評(píng)論

# re: 基類(lèi)角色之對(duì)象管理器 2009-07-02 14:10 Kevin Lynx

本質(zhì)上就是通過(guò)讓其他模塊不保存這個(gè)object的直接指針,而是一個(gè)ID,然后通過(guò)一個(gè)manager由這個(gè)ID而獲取到這個(gè)object*,是可以有效減少指針無(wú)效的問(wèn)題。但是,面對(duì)的最大問(wèn)題就在于這個(gè)查找過(guò)程。你這里用的是std::map,我們用的是stdext::hash_map,我自己都覺(jué)得有點(diǎn)速度問(wèn)題。希望能在這個(gè)問(wèn)題上找到良好的解決辦法。:)  回復(fù)  更多評(píng)論   

# re: 基類(lèi)角色之對(duì)象管理器 2009-07-02 15:20 飯中淹

我一般是用一個(gè)ID管理器來(lái)解決這個(gè)問(wèn)題。
采用直接索引的辦法。對(duì)于重復(fù)的ID使用一個(gè)順序增加的KEY來(lái)做驗(yàn)證。

也就是一個(gè)ID,一個(gè)KEY來(lái)索引一個(gè)對(duì)象。

之前還有一個(gè)方法,就是采用一種LINK對(duì)象,LINK對(duì)象能鏈接到另外一個(gè)LINK對(duì)象,并互換HOST指針,并且有主從關(guān)系的LINK方式。
當(dāng)一個(gè)LINK對(duì)象銷(xiāo)毀的時(shí)候,會(huì)通知所有已經(jīng)連接的LINK對(duì)象放棄自己的HOST指針。這樣的話,別的對(duì)象里面保存的指針會(huì)隨著原始對(duì)象的銷(xiāo)毀而自動(dòng)清空。
  回復(fù)  更多評(píng)論   

# re: 基類(lèi)角色之對(duì)象管理器 2009-07-02 17:24 Kevin Lynx

雖然不是很明白飯中淹關(guān)于LINK對(duì)象的意思,但是我獲得了靈感,寫(xiě)了下面的代碼:
///
///
///
#include <stdio.h>
#include <list>

class ref_op
{
public:
virtual void be_null() { }
};

typedef std::list<ref_op*> RefOpList;
class ref_base
{
public:
~ref_base()
{
clear_ops();
}

void add_ref( ref_op *op )
{
_oplist.push_back( op );
}

void clear_ops()
{
for( RefOpList::const_iterator it = _oplist.begin();
it != _oplist.end(); ++ it )
{
(*it)->be_null();
}
}

private:
RefOpList _oplist;
};

template <typename _Tp>
class auto_null : public ref_op
{
public:
void fetch( _Tp *t )
{
_t = t;
t->add_ref( this );
}
auto_null<_Tp> &operator = ( _Tp *t )
{
fetch( t );
return *this;
}
void be_null()
{
_t = 0;
}
operator _Tp*()
{
return _t;
}
private:
_Tp *_t;
};

//////////////////////////////////////////////////////////////////////////////
class CMonster : public ref_base
{
};

class CMonsterAI
{
public:
void SetOwner( CMonster *pOwner )
{
m_Owner = pOwner;
}

void Test()
{
if( (CMonster*)m_Owner == NULL )
{
printf( "The owner is null.\n" );
}
else
{
printf( "The owner is NOT null.\n" );
}
}
private:
auto_null<CMonster> m_Owner;
};

int main()
{
CMonster *pMonster = new CMonster();
CMonsterAI *pAI = new CMonsterAI();
pAI->SetOwner( pMonster );
pAI->Test();
delete pMonster;
pAI->Test();
delete pAI;
return 0;
}

CMonster內(nèi)部會(huì)保存一個(gè)CMonster的指針,當(dāng)CMonster被刪除掉時(shí),會(huì)自動(dòng)更新CMonsterAI內(nèi)部的CMonster“指針”為NULL。接口比較簡(jiǎn)單:1)在會(huì)被引用(即被其他對(duì)象保存其指針)的類(lèi)設(shè)計(jì)中派生(或組合)ref_base類(lèi),ref_base類(lèi)簡(jiǎn)單來(lái)說(shuō)就是保存一個(gè)引用類(lèi)對(duì)象列表,通過(guò)基類(lèi)ref_op可以使得ref_base保存不同類(lèi)型的引用類(lèi)(例如CMonsterAI可以保存CMonster, CRegion也可以保存CMonster),ref_base在析構(gòu)時(shí)自動(dòng)回調(diào)引用類(lèi)對(duì)象的be_null函數(shù),be_null函數(shù)在auto_null類(lèi)中自動(dòng)把CMonster*設(shè)置為NULL。
通過(guò)重載operator=,使得SetOwner函數(shù)里不需要做其他操作(看起來(lái))。
  回復(fù)  更多評(píng)論   

# re: 基類(lèi)角色之對(duì)象管理器 2009-07-02 18:02 李現(xiàn)民

@飯中淹
對(duì)于“也就是一個(gè)ID,一個(gè)KEY來(lái)索引一個(gè)對(duì)象”這種辦法,我想可以通過(guò)增加ID(我我的實(shí)現(xiàn)代碼是是index_t)的長(zhǎng)度來(lái)代替,即定義:
typedef __int64 index_t; // 索引類(lèi)型

我認(rèn)為效果是等價(jià)的。

  回復(fù)  更多評(píng)論   

# re: 基類(lèi)角色之對(duì)象管理器 2009-07-02 18:05 李現(xiàn)民

@Kevin Lynx
我感覺(jué)那個(gè)link的意思有點(diǎn)像觀察者observer,即:當(dāng)一個(gè)對(duì)象銷(xiāo)毀時(shí)通知所有對(duì)其感興趣的對(duì)象,告訴它們:我死了,不要再來(lái)找我

但這種方式,我個(gè)人覺(jué)得有點(diǎn)復(fù)雜了  回復(fù)  更多評(píng)論   

# re: 基類(lèi)角色之對(duì)象管理器[未登錄](méi) 2009-07-02 21:51 塵埃

id用兩個(gè)部分組合而成:數(shù)組索引、累加,64位或許應(yīng)該夠了,且短時(shí)間內(nèi)很難發(fā)生碰撞吧。數(shù)組索引用來(lái)立即找出對(duì)應(yīng)對(duì)象,再用累加部分驗(yàn)證一下,實(shí)在不放心把累加部分?jǐn)U大到64位,對(duì)于客戶(hù)端來(lái)說(shuō)怎么都該夠了;)  回復(fù)  更多評(píng)論   

# re: 基類(lèi)角色之對(duì)象管理器 2009-07-03 09:55 zuhd

@Kevin Lynx
兄弟這段代碼很經(jīng)典啊,學(xué)習(xí)了!   回復(fù)  更多評(píng)論   

# re: 基類(lèi)角色之對(duì)象管理器 2009-07-03 10:16 飯中淹

@Kevin Lynx
跟這個(gè)差不多。不過(guò)不用繼承

xLink<Host, Target>


xLink.link( xLink<Target, Host> & lnk );
xLink.disLinkAll();

*
&
是重載的。
返回Host
  回復(fù)  更多評(píng)論   

# re: 基類(lèi)角色之對(duì)象管理器 2009-07-03 10:18 飯中淹

@李現(xiàn)民
確實(shí)是類(lèi)似觀察者這種,不過(guò)更直接了
  回復(fù)  更多評(píng)論   

# re: 基類(lèi)角色之對(duì)象管理器 2009-07-03 12:03 李現(xiàn)民

@Kevin Lynx
今天我看了下,你這種實(shí)現(xiàn)方式就已經(jīng)是觀察者了,其中CMonster 就是被觀察的(observable),而所有使用auto_null<>對(duì)象的都是觀察者(observer), 這種實(shí)現(xiàn)方法我覺(jué)得至少有兩個(gè)缺點(diǎn):
1,會(huì)強(qiáng)制observable都使用ref_base 基類(lèi),同時(shí)observer需要使用auto_null<>去封裝對(duì)象,因此即使新設(shè)計(jì)的類(lèi)可以使用auto_null,但舊有的、需要被觀察的類(lèi)無(wú)法自動(dòng)成為observable,因?yàn)樗鼈儾辉^承ref_base
2,使用auto_null對(duì)象時(shí),IDE的智能通常無(wú)法診測(cè)出具體對(duì)象(CMonster)內(nèi)部的函數(shù)名,只能診測(cè)出像fetch()這些  回復(fù)  更多評(píng)論   

# re: 基類(lèi)角色之對(duì)象管理器 2009-07-04 16:29 99網(wǎng)上書(shū)店

確實(shí)是類(lèi)似觀察者這種,不過(guò)更直接了  回復(fù)  更多評(píng)論   

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            欧美在线在线| 久久久久99| 国产精品二区三区四区| 欧美国产在线观看| 欧美日韩成人综合| 国产精品激情偷乱一区二区∴| 欧美日本在线| 国产精品卡一卡二| 国产亚洲亚洲| 亚洲国产精品电影| 亚洲伦伦在线| 性欧美长视频| 欧美风情在线观看| 亚洲第一毛片| 亚洲精品永久免费| 亚洲欧美视频在线观看| 久久国产天堂福利天堂| 欧美激情a∨在线视频播放| 欧美色欧美亚洲高清在线视频| 国产欧美日本| 亚洲精品字幕| 快射av在线播放一区| 一本色道88久久加勒比精品| 久久久精品一品道一区| 国产精品久久久| 亚洲精品日韩欧美| 久久精彩视频| 99精品欧美一区| 国产精品久久久久婷婷| 一区在线视频| 亚洲欧美中文日韩在线| 欧美成人午夜| 久久国产精品一区二区三区四区| 欧美日韩伦理在线免费| 影音先锋久久久| 欧美一区二区三区免费观看| 亚洲日产国产精品| 久久网站免费| 国产一区二区三区黄视频| 一区二区三区精品国产| 免费视频一区二区三区在线观看| 亚洲天堂av综合网| 欧美日本亚洲韩国国产| 亚洲福利精品| 久久综合影视| 欧美综合激情网| 国产一区美女| 久久国产一区二区| 亚洲欧美网站| 国产欧美韩日| 欧美一区二区免费视频| 亚洲欧洲在线视频| 欧美国产精品久久| 亚洲人妖在线| 亚洲国产精品欧美一二99| 麻豆精品网站| 亚洲激情在线观看| 欧美国产精品专区| 欧美www视频在线观看| 亚洲国产美国国产综合一区二区| 久久久久网址| 久久久久久久性| 怡红院av一区二区三区| 久久精品一区四区| 久久精品视频导航| 曰本成人黄色| 亚洲黄色性网站| 欧美日韩中字| 欧美影院在线| 久久精品中文| 亚洲人成网站在线播| 亚洲精品影院在线观看| 国产精品推荐精品| 久久久久久久久久久成人| 久久久99久久精品女同性| 亚洲国产成人久久综合| 亚洲日本久久| 欧美日韩一区二区视频在线观看| 亚洲永久免费| 久久久99精品免费观看不卡| 亚洲激情av在线| 99国产精品久久久久老师 | 一区二区三区免费看| 国产精品毛片大码女人| 久久久国产精品亚洲一区 | 亚洲国产成人不卡| 老鸭窝91久久精品色噜噜导演| 欧美一区二区久久久| 国产在线日韩| 亚洲人成7777| 国产一区二三区| 日韩午夜精品| 国产伦精品一区二区三区视频黑人| 中国女人久久久| 日韩午夜电影av| 欧美一区二区在线观看| 日韩视频一区二区三区| 国产精品久久999| 欧美gay视频| 欧美先锋影音| 欧美成人午夜免费视在线看片 | 亚洲丶国产丶欧美一区二区三区| 亚洲国产精品电影| 国产欧美一区二区三区久久人妖| 欧美 日韩 国产在线| 国产精品av一区二区| 猫咪成人在线观看| 欧美三级乱人伦电影| 久久亚洲精品伦理| 国产精品久久久久久久久久妞妞| 欧美大片91| 国产一区在线看| 亚洲在线中文字幕| 在线亚洲一区二区| 欧美xx69| 亚洲第一搞黄网站| 亚洲第一天堂av| 欧美一区二区三区四区在线观看| aa亚洲婷婷| 欧美大片免费观看在线观看网站推荐| 久久成年人视频| 国产精品对白刺激久久久| 亚洲国产精品成人久久综合一区| 国产在线播放一区二区三区| 中日韩视频在线观看| 日韩视频三区| 美女国内精品自产拍在线播放| 久久成人久久爱| 国产日本欧美一区二区| 亚洲性xxxx| 亚洲男人影院| 国产精品国产三级国产普通话99| 亚洲欧洲精品天堂一级| 亚洲第一视频| 鲁大师影院一区二区三区| 久久综合色天天久久综合图片| 国产亚洲网站| 久久久人人人| 欧美mv日韩mv国产网站| 亚洲国产精品99久久久久久久久| 久久久久久久激情视频| 久久亚洲精品一区二区| 亚洲精品久久在线| 香蕉精品999视频一区二区 | 狂野欧美激情性xxxx| 国产在线视频欧美| 久久精品国产免费观看| 麻豆91精品| 亚洲电影免费在线观看| 麻豆精品传媒视频| 亚洲精一区二区三区| 亚洲性视频网站| 国产精品私拍pans大尺度在线| 香蕉久久夜色精品| 免费观看亚洲视频大全| 亚洲激情午夜| 国产精品激情av在线播放| 午夜精品福利在线观看| 久久久亚洲精品一区二区三区| 伊人久久大香线蕉综合热线| 欧美成人免费一级人片100| 亚洲毛片在线观看| 午夜在线电影亚洲一区| 国产亚洲欧美一区| 亚洲欧美国产一区二区三区| 国产亚洲精品bt天堂精选| 香港久久久电影| 久久亚洲精品伦理| 国产精品99久久99久久久二8| 国产精品美女黄网| 欧美一区二区三区在线| 亚洲国产网站| 久久精品日韩欧美| 亚洲日本免费电影| 国产精品久久久久久久久果冻传媒 | 亚洲一区欧美激情| 久久精品一区蜜桃臀影院 | 欧美高清在线| 亚洲欧美激情视频| 尤物视频一区二区| 欧美日韩国产在线看| 欧美一级视频免费在线观看| 亚洲大胆人体在线| 欧美在线一区二区| 99热精品在线| 国内精品伊人久久久久av一坑| 欧美顶级大胆免费视频| 午夜精品一区二区三区在线播放 | 久久久久这里只有精品| 一个色综合av| 91久久一区二区| 免费一级欧美片在线观看| 性高湖久久久久久久久| 一区二区高清| 亚洲精品一区二区三区樱花| 激情欧美日韩| 国产欧美日韩另类一区 | 亚洲精品国产精品乱码不99| 国产婷婷一区二区| 国产精品久久久久99|