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

woaidongmao

文章均收錄自他人博客,但不喜標題前加-[轉貼],因其丑陋,見諒!~
隨筆 - 1469, 文章 - 0, 評論 - 661, 引用 - 0
數據加載中……

讓vector析構時不釋放內存

摘要:通過指定allocator的辦法控制vector的內存釋放。

關鍵字:allocator vector 內存釋放


經常有這樣的情況,一個函數要返回一個不定長的數組。一般情況下,直接在函數里面動態分配內存就好了。不過很多時候,往往涉及到更復雜的操作,如插入/刪除元素或是出棧/入棧什么的,這時使用STLvector就是最好的選擇了。但這種情況下,vector有一個不方便的地方,就是它的析構函數會釋放內存。這當然是正確的行為,但是這樣一來,我們的代碼就變成這樣了:

T * foo()
{
    std::vector<T > aa;

    ...//
使用aa

    T *buf = (T*)malloc(aa.size()*sizeof(T));

    memcpy(buf,&aa[0],aa.size()*sizeof(T));

    return buf;
}

其中T是一個POD類型.

在函數結尾的時候,我們無法直接返回 &aa[0],雖然那就是我們要的數組。因為aa是一個類,它的析構函數在函數返回前必須被調用,函數結束后,對應的內存已經無效。所以在返回之前,要將它的內存復制一次.

當然,你可以繞過這個問題,只要將aa作為函數的參數,而不是局部變量就可以了。不過我想要的是直接的解決方法,而不是繞過去。雖然你現在也許可以繞過去,但你遲早碰到繞不過去的情況,比如若函數的規格已經確定不能更改呢?

這里要解決的問題就是"如何使vector析構時不釋放內存"。看起來很奇怪的要求,但實際上有時候卻是合理的。解決辦法當然有,別忘了,stl有著極好的彈性。在使用vector的時候,本來是有兩個參數的,上面的例子只使用了一個類型參數,第二個參數就是解決辦法所在了。

vector
的第二個參數是指定vector使用的內存分配類allocator。只要我們實現自己的allocator,就可以讓vector按找我們的意思分配釋放內存。allocator的規格可以在VC的頭文件...vc98\include\xmemory里面看到。這里插一句,本來allocator的規格在stl中是有規定的,不過stl有好多版本的實現,其中有些版本就未必遵循stl的標準規則。比如vc6使用了P.J.Plauger版本,就沒有完全遵循標準規格。所以我們必須搞清楚相應的規格才行。

下面就是我改寫的一個allocator,為了與標準的區別開來,將它包含在名字空間lhf中。

namespace lhf
{
    int MemKeep_flag = 0;

    void setMemKeep(bool flag){ MemKeep_flag=flag; }

    template<class _Ty>//
自定義的內存管理,目的是不釋放vector的內存.
    class allocator
    {   
        public:
        typedef size_t size_type;
        typedef ptrdiff_t difference_type;
        typedef _Ty _FARQ * pointer;
        typedef const _Ty _FARQ * const_pointer;
        typedef _Ty _FARQ& reference;
        typedef const _Ty _FARQ& const_reference;
        typedef _Ty value_type;

        pointer address(reference _X) const { return (&_X); }
        const_pointer address(const_reference _X) const {return (&_X); }

        pointer allocate(size_type _N, const void *){ return (pointer)malloc(_N*sizeof(_Ty));}
        char _FARQ *_Charalloc(size_type _N){ return (char _FARQ *)malloc(_N); }
        void deallocate(void _FARQ *_P, size_type){ if(!MemKeep_flag) free(_P);}

        void construct(pointer _P, const _Ty& _V){ new ((void _FARQ *)_P) (_Ty)(_V); }
        void destroy(pointer _P){ _P; (_P)->~_Ty(); }

        _SIZT max_size() const{_SIZT _N = (_SIZT)(-1) / sizeof (_Ty); return (0 < _N ? _N : 1); }
    };
}

上面的代碼中,主要就是增加了一個全局變量MemKeep_flag和函數setMemKeep,來控制vector是否能釋放掉內存。如果MemKeep_flag為假,則vector可以象平時一樣,正常釋放內存。否則,vector就不能釋放內存,即使析構函數被調用了。

現在,上面的函數就可以這樣使用vector.

T * foo()
{
    T *buf = 0;

    {
        std::vector<T,lhf::allocator<T> > aa;

        ...//
使用aa

        buf = (T*)&aa[0];
        lhf::setMemKeep(true);//
禁止釋放內存
    }

    lhf::setMemKeep(false);//
恢復釋放內存

    return buf;
}

注意上面的兩次setMemKeep的調用,其位置很重要(在兩次調用之間必須包含析構函數的調用)。另外,將aa的定義放在一個大括號內也很重要,這決定了aa何時調用析構函數。

談一個東西只說好處而不說壞處是可恥的,這容易讓人想起政客或者商家們的推銷。首先,前面已經說過,各版本的STL實現不盡相同,上面的allocator也許無法與某些stl版本搭配使用,或是可以使用,但反而大大降低了效率。其次,注意到上面的代碼中使用了全局變量MemKeep_flag,我們知道,象這樣使用全局變量的代碼是無法經受多線程考驗的。比如,一個線程調用了setMemKeep(true),在調用setMemKeep(false)之前就切換到了另外一個線程。這下好戲上演了,在后一個線程中,vector將無法釋放任何內存!

再次提醒一下,如果你要在代碼中使用類似的花招,千萬要記住,setMemKeep(true)setMemKeep(false)要成對調用,中間的間隔越小越好,并確保不會涉及到多線程的問題。切記,切記!

最后祝編程愉快!

最近將上面的代碼移植到vs2005下,居然過不了。一看,原來allocator的 接口定義改了,shit。唉,看來完全自己實現一個allocator 是靠不住了。只好派生一個算了,下面是新的代碼:

namespace lhf
{
int MemKeep_flag = 0;

void setMemKeep(bool flag){ MemKeep_flag=flag; }

template<class _Ty>//自定義的內存管理,目的是不釋放vector的內存.
class allocator : public std::allocator<class _Ty>
{
public:
  void deallocate(pointer _Ptr, size_type){ if(!MemKeep_flag) __super::deallocate(__Ptr);}
};
}

 

posted on 2011-03-29 10:23 肥仔 閱讀(3234) 評論(2)  編輯 收藏 引用 所屬分類: Boost & STL

評論

# re: 讓vector析構時不釋放內存[未登錄]  回復  更多評論   

在我看來這與直接new沒有什么不同,雖然被allocator包裝了
用這樣的函數已經不是小心不小心的問題了
2011-03-29 15:11 | vincent

# re: 讓vector析構時不釋放內存  回復  更多評論   

這樣雖然vector不釋放內存,可是里面所有的T元素都被析構了。這種東西有什么意義呢?
2012-06-22 14:39 | 徐大俠
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美电影在线免费观看网站| 国模一区二区三区| 久久不射中文字幕| 国产精品女人网站| 亚洲欧美国产另类| 麻豆国产va免费精品高清在线| 在线观看日韩精品| 国产精品永久免费观看| 国产日产亚洲精品| 欧美一区二区三区在线播放| 欧美日韩中文精品| 亚洲欧美综合国产精品一区| 麻豆久久久9性大片| 一本色道久久综合狠狠躁篇的优点| 欧美私人啪啪vps| 久久久久国色av免费看影院| 亚洲第一免费播放区| 国产精品乱码人人做人人爱| 久久久欧美一区二区| 99精品国产在热久久| 欧美成人免费在线| 欧美一区二区在线免费观看| 欧美在线首页| 亚洲精品久久久久久一区二区| 欧美黄污视频| 亚洲欧美国产一区二区三区| 美脚丝袜一区二区三区在线观看 | 国产欧美日韩三级| 揄拍成人国产精品视频| 一区二区三区|亚洲午夜| 欧美一区二区三区另类| 欧美99久久| 亚洲欧美综合国产精品一区| 欧美精品啪啪| 亚洲国产专区| 久久久久久久综合| 国产精品99久久99久久久二8 | 国产日韩欧美在线视频观看| 久久狠狠亚洲综合| 欧美视频中文字幕| 亚洲激情在线激情| 欧美一区二区黄色| 日韩午夜av电影| 欧美成人免费小视频| 国产一区二区三区免费不卡| 亚洲午夜一区| 91久久国产综合久久91精品网站| 新狼窝色av性久久久久久| 欧美日韩国产黄| 亚洲国产精品综合| 麻豆精品网站| 欧美一区二区三区四区在线观看地址| 欧美精品日韩综合在线| 亚洲国产欧美日韩精品| 美女脱光内衣内裤视频久久影院 | 亚洲人体大胆视频| 久久深夜福利免费观看| 国产日韩欧美综合| 午夜精品久久久久久| 日韩午夜在线播放| 欧美激情综合网| 亚洲三级视频| 亚洲欧洲一区| 免费成人网www| 亚洲二区在线视频| 媚黑女一区二区| 久久青草欧美一区二区三区| 国产三区精品| 久久日韩精品| 久久伊伊香蕉| 亚洲精品少妇网址| 亚洲精品中文字幕女同| 欧美日韩免费一区| 亚洲影院一区| 先锋影音国产一区| 国内精品模特av私拍在线观看| 欧美在线视频不卡| 亚洲欧美中日韩| 国产乱子伦一区二区三区国色天香| 亚洲免费在线观看| 欧美怡红院视频一区二区三区| 激情丁香综合| 欧美成在线视频| 欧美激情一二区| 亚洲天堂网在线观看| 一本色道久久综合亚洲二区三区| 国产精品美女久久| 欧美一级一区| 久久精品国产亚洲5555| 黑人操亚洲美女惩罚| 麻豆成人综合网| 欧美二区在线观看| 亚洲一区三区在线观看| 亚洲第一综合天堂另类专| 欧美一区二区免费| 欧美日韩一区高清| 亚洲国产高清自拍| 久久久亚洲精品一区二区三区 | 亚洲三级国产| 久久精品中文| 久久久久久久一区二区| 曰韩精品一区二区| 91久久精品国产91久久性色tv| 欧美另类99xxxxx| 欧美一区网站| 欧美精品激情| 久久亚洲视频| 国产精品www.| 欧美国产91| 国产麻豆91精品| 亚洲欧洲精品一区| 激情国产一区二区| 亚洲一区日韩在线| 亚洲美女一区| 久久久久久一区| 亚洲欧美另类在线| 欧美精品1区| 免费成人你懂的| 国产日韩av高清| 99在线精品观看| 亚洲国产精品一区二区尤物区| 一区二区三区欧美在线| 亚洲激情视频| 久久久精品一品道一区| 亚欧美中日韩视频| 欧美日韩在线不卡| 亚洲电影专区| 亚洲国产欧美国产综合一区| 久久久国产精品亚洲一区| 亚洲欧美另类在线观看| 欧美日韩国产色综合一二三四| 欧美高清一区| 亚洲欧洲一区二区三区久久| 性高湖久久久久久久久| 欧美亚洲三级| 欧美香蕉大胸在线视频观看| 亚洲日本久久| 一本色道久久综合一区| 欧美精品v国产精品v日韩精品 | 国产精品一区三区| 国内在线观看一区二区三区| 亚洲视频二区| 亚洲伊人第一页| 国产精品成人一区二区网站软件 | 国产中文一区| 欧美一级理论性理论a| 欧美一区二区免费视频| 国产精品一区二区男女羞羞无遮挡 | 久久婷婷蜜乳一本欲蜜臀| 国产亚洲日本欧美韩国| 亚洲在线一区二区三区| 午夜日本精品| 国产日韩欧美不卡| 欧美在线网址| 欧美激情精品久久久| 亚洲精选一区二区| 欧美日韩无遮挡| 亚洲先锋成人| 久久久精品日韩欧美| 黄色精品一区| 欧美激情一区在线观看| 夜夜嗨av一区二区三区网页| 午夜视频久久久| 精品99一区二区| 欧美va亚洲va国产综合| 99re8这里有精品热视频免费 | 国产欧美一区二区色老头| 欧美怡红院视频| 亚洲第一精品福利| 亚洲午夜国产一区99re久久| 国产伦理一区| 老司机aⅴ在线精品导航| 亚洲人成欧美中文字幕| 亚洲——在线| 好男人免费精品视频| 欧美激情综合网| 亚洲亚洲精品三区日韩精品在线视频| 久久精品人人爽| 亚洲人成在线影院| 国产伦精品一区二区三区免费迷 | 欧美一区二区免费| 亚洲区中文字幕| 久久亚洲视频| 午夜精品福利电影| 在线日韩中文字幕| 国产精品午夜在线观看| 欧美不卡视频| 久久久91精品国产| 一级日韩一区在线观看| 欧美aa在线视频| 性色av一区二区三区在线观看| 亚洲人成网在线播放| 国产欧美精品日韩| 在线中文字幕不卡| 久久久精品午夜少妇| 午夜精品视频一区| 一本大道av伊人久久综合| 国内综合精品午夜久久资源| 欧美午夜不卡视频| 男人的天堂亚洲在线|