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

woaidongmao

文章均收錄自他人博客,但不喜標(biāo)題前加-[轉(zhuǎn)貼],因其丑陋,見諒!~
隨筆 - 1469, 文章 - 0, 評(píng)論 - 661, 引用 - 0
數(shù)據(jù)加載中……

C++ 標(biāo)準(zhǔn)庫中的 allocator 是多余的

我認(rèn)為C++allocator是依賴注入的一次失敗的嘗試。

C/C++
里的內(nèi)存分配和釋放是個(gè)重要的事情,我同意,在寫library的時(shí)候,除了默認(rèn)使用 malloc/free,還應(yīng)該允許用戶指定使用內(nèi)存分配的函數(shù)。用現(xiàn)在的話說,如果library依賴于內(nèi)存分配與釋放,就應(yīng)該允許用戶注入這種依賴。我看到有些C library是支持這個(gè)的,可以在初始化時(shí)傳入兩個(gè)函數(shù)指針,指向內(nèi)存分配和釋放的函數(shù)。


問題是,allocator是模板參數(shù),而不是構(gòu)造函數(shù)的參數(shù)。這意味著


1.
由于不能從構(gòu)造函數(shù)傳入allocator,那么每種類型的allocator必須是全局唯一的(Singleton)。無論SGI 的內(nèi)存池(稱為PoolAlloc),還是簡(jiǎn)單的new wrapper(稱為NewAlloc)都只從一個(gè)地方(region)搞到內(nèi)存,這大大限制了其使用。


2. allocator
vector類型的一部分,vector<string, PoolAlloc> vector<string, NewAlloc> 是兩個(gè)類型,不可相互替換。這不僅暴露了實(shí)現(xiàn),還暴露到了類型上,恐怕沒有比這更糟糕的了。


下面舉例說明,


對(duì)于1,假設(shè)我有一個(gè)任務(wù)(假設(shè)是parse),需要分配很多小塊內(nèi)存,總?cè)萘坎怀^20M。為了防止內(nèi)存泄露及避免內(nèi)存碎片,我希望在任務(wù)開始時(shí),先從系統(tǒng)拿到20M內(nèi)存,供這個(gè)任務(wù)使用(parse里分配內(nèi)存只需要改一個(gè)指針,釋放內(nèi)存是空操作),等任務(wù)完成后,我一次性釋放這20M內(nèi)存,這樣既高效又安全。然而C++allocator并不能幫我實(shí)現(xiàn)這一點(diǎn),因?yàn)樗侨值摹N也荒芴鎿Q全局的 allocator,因?yàn)槟菚?huì)影響其他線程。也不能在運(yùn)行時(shí)指定某個(gè)vector<string>用哪種allocator,因?yàn)轭愋褪蔷幾g時(shí)確定的。


對(duì)于2,如果我想寫一個(gè)普通的以vector<string>為參數(shù)的函數(shù),這不難


void process(vector<string>& records);

由于vector<string, PoolAlloc>vector<string, NewAlloc>類型不同,process只能接受一種。

但這完全沒道理,我不過想訪問一個(gè)vector<string>,根本不關(guān)心它的內(nèi)存是怎么分配的,卻被什么鬼東西allocator擋在了門外。


我要么提供重載:
void process(vector<string, NewAlloc>& records);
void process(vector<string, PoolAlloc>& records);

要么改寫成模板:

template<typename Alloc>
void process(vector<string, Alloc>& records);
//
(同理可知,如果在一個(gè)程序里使用多種allocator,那么所有涉及標(biāo)準(zhǔn)庫容器的函數(shù)都必須改寫為函數(shù)模板)



無論哪種"解決辦法"都會(huì)導(dǎo)致代碼膨脹,而且給標(biāo)準(zhǔn)庫的使用者帶來完全不必要的負(fù)擔(dān)。


更糟糕的是,allocator給程序庫的作者也帶來了不必要的負(fù)擔(dān)。如果想把process(vector<string>& records)放到某個(gè)library中,那么為了適應(yīng)不同的allocator,必須把函數(shù)定義放在頭文件里(因?yàn)檫@是個(gè)函數(shù)模板)。明明是針對(duì)一個(gè)固定類型(vector of string)的函數(shù),卻不得不寫成函數(shù)模板,把實(shí)現(xiàn)細(xì)節(jié)暴露在頭文件里,讓每個(gè)用戶都去編譯一遍,這真是完全沒道理。


根據(jù)以上的分析,基本上不可能在一個(gè)程序里混用多種allocator,既然一個(gè)程序只能有一種allocator,那為什么還要放到每個(gè)容器的模板參數(shù)里呢?提供一個(gè)全局的鉤子不就行了嘛?

相反,shared_ptr就只有一個(gè)模板參數(shù)T,而他同樣可以指定allocator----在構(gòu)造時(shí)傳入。


現(xiàn)在看來,vector(以及其他標(biāo)準(zhǔn)庫容器)與其增加一個(gè)Alloc模板參數(shù),不如在構(gòu)造時(shí)傳入兩個(gè)函數(shù)指針,一個(gè)allocate,一個(gè)deallocate,定制的效果也一樣。只不過這么做會(huì)讓標(biāo)準(zhǔn)委員會(huì)的人覺得不夠GP,很可能被拍掉。

總而言之,allocator并不能達(dá)到精確控制(定制)內(nèi)存分配釋放的目的,雖然它名以上是為此而生的。雖然在歷史上可能有過正面效果(封裝 far / near pointer),但現(xiàn)在它無疑就是個(gè)累贅。


allocator
就跟 IOStream Locale/Facetauto_ptr valarray 一樣,成為C++標(biāo)準(zhǔn)一直要背負(fù)的歷史包袱。

有關(guān)問題:

> allocator是個(gè)類模板啊... ...其實(shí)例化要依賴于容器所關(guān)聯(lián)的數(shù)據(jù)類型啊,怎么從構(gòu)造函數(shù)傳入?

allocator
一開始就不應(yīng)該是個(gè)類模板,它只要提供void* alloc(size_t num_bytes) void free(void*) 這兩個(gè)功能,就能讓容器正確實(shí)現(xiàn)。畢竟在C++里,分配內(nèi)存、構(gòu)造對(duì)象、析構(gòu)對(duì)象、釋放內(nèi)存這四步是可以獨(dú)立開的,allocator 只要管好第1步和第4步就好了。

就算allocator是個(gè)未知類型,它也可以從構(gòu)造函數(shù)傳入,而無需作為容器本身的模板參數(shù)。同樣的技術(shù)在shared_ptr里用過,Scott Meyers 也總結(jié)過: http://www.artima.com/cppsource/top_cpp_aha_moments.html (第5條)

>
而且何以有所謂 allocatorsingleton...?
每個(gè)容器有一個(gè)allocator的實(shí)例,從這個(gè)意義上講,allocator不是Singleton。但是,由于不能從容器的構(gòu)造函數(shù)傳入allocator,那么容器在實(shí)例化allocator時(shí)只能用默認(rèn)構(gòu)造函數(shù)。這等于是讓容器自己持有的allocator變成一個(gè)轉(zhuǎn)發(fā)器,要么轉(zhuǎn)為調(diào)用全局的malloc/free,要么轉(zhuǎn)為調(diào)用某個(gè)全局的allocator(比如 SGI 的 內(nèi)存池)。從這個(gè)意義上講,每種*類型*allocator(不是每個(gè)allocator實(shí)體)只能從一塊地方分配內(nèi)存,所以是個(gè)Singleton


>
如果同一個(gè)實(shí)例化的容器類的不同對(duì)象如果使用不同的allocator,如何保證swap的有效和正確?

shared_ptr
是如何做到的?shared_ptr 能從構(gòu)造函數(shù)接受DeleterAlloctor,而且實(shí)現(xiàn)了swap()。

> 但還需探討一下:我們能否改進(jìn)allocator的設(shè)計(jì)?究竟這個(gè)問題是mission impossible,或者只是Stepanov沒有找到最佳的設(shè)計(jì)而已?

很難說。自然,我們可以把allocator變成容器的成員而不是模板類型參數(shù)。

或者干脆把allocator定義成一個(gè)abstract base class,然后在容器里放allocator的指針(這么做肯定在標(biāo)準(zhǔn)委員會(huì)通不過,那幫人有虛函數(shù)恐懼癥)。但是這樣做了之后,馬上遇到一個(gè)問題:allocator的生命期誰來控制,是不是在容器析構(gòu)的時(shí)候同時(shí)析構(gòu)allocator?如果這樣,一個(gè)allocator就只能給一個(gè)容器用。如果不析構(gòu),那么allocator又由誰來釋放呢?

或者按照shared_ptr的做法,allocator是未知類型的,具有值語義,但不必作為容器的模板參數(shù)(allocator實(shí)際上是容器的成員,但是其類型及大小未知,所以用了一個(gè)trick),這樣生命期的問題倒是解決了。只不過allocator沒辦法內(nèi)聯(lián),(甚至要通過虛函數(shù)來調(diào)用),很可能會(huì)被標(biāo)準(zhǔn)委員會(huì)拍死。

另外一個(gè)更為重要的問題,allocator沒有傳遞性。比如vector<string> vs,如果 vs 用了我的MyAlloc,我希望它持有的string也用MyAlloc了分配內(nèi)存,這樣我對(duì)vs對(duì)象涉及的全部?jī)?nèi)存管理都能有所控制。但是很可惜除了自己寫vector,沒辦法把MyAlloc傳遞給vector里的string。(如果把MyAlloc作為string類型的模板參數(shù),就又回到原來的老路上去了。)

posted on 2011-03-29 10:16 肥仔 閱讀(1883) 評(píng)論(5)  編輯 收藏 引用 所屬分類: Boost & STL

評(píng)論

# re: C++ 標(biāo)準(zhǔn)庫中的 allocator 是多余的[未登錄]  回復(fù)  更多評(píng)論   

allocator的確不盡人意,也確實(shí)像你說的那樣Singleton,難道m(xù)alloc/free和new/delete就不Singleton,我看未必吧。管理內(nèi)存是件很復(fù)雜的事情,無論allocator還是malloc/free和new/delete無非都是一wrapper而已。如果CPU個(gè)數(shù)多了,它們都會(huì)死掉,甚至PC系統(tǒng)內(nèi)存管理相關(guān)的API都會(huì)死掉,因此你不應(yīng)該指責(zé)allocator。如果想設(shè)計(jì)一個(gè)高并發(fā)獲得很好加速比,有效防止blowup和false sharing,且低的overhead,那是一件很困難的事情,目前沒有一個(gè)這么絕對(duì)理想的。我不愿意聽什么大牛二牛,我只知道Win2008和所有耳聞的Unix版本目前都做不到這么理想。

STL容器之所以有個(gè)Alloc,是希望用戶在必要時(shí)自定義allocator,不同的allocator怎么可能互換?一個(gè)進(jìn)程內(nèi)不同的內(nèi)存地址還能互換嗎?這豈不荒唐,即使允許重新映射的別的物理地址,那是操作系統(tǒng)的事情,allocator能有這本事豈不笑談,那還要操作系統(tǒng)虛擬內(nèi)存管理器干啥?

shared_ptr有啥好的?除了performance問題和更大的內(nèi)存overhead,還能有啥好處?但是無可厚非,在速度和內(nèi)存要求不高的場(chǎng)合,或者數(shù)據(jù)量很小時(shí),用它也未嘗不可。我們總得尊重人權(quán)吧!
2011-08-18 22:58 | Chipset

# re: C++ 標(biāo)準(zhǔn)庫中的 allocator 是多余的  回復(fù)  更多評(píng)論   

硬傷:不能從構(gòu)造函數(shù)傳入allocator。

自己看下能不能傳。
2012-05-02 12:22 | nous

# re: C++ 標(biāo)準(zhǔn)庫中的 allocator 是多余的  回復(fù)  更多評(píng)論   

怎么加關(guān)注啊
2013-01-08 13:20 | hxs

# re: C++ 標(biāo)準(zhǔn)庫中的 allocator 是多余的  回復(fù)  更多評(píng)論   

stl容器在原則上是支持stateful allocator的,它調(diào)用allocate()是通過點(diǎn)算符而不是::算符,這樣每個(gè)對(duì)象都有自己的allocator,雖然類型上是一樣的。
2013-05-15 03:51 | wxy

# re: C++ 標(biāo)準(zhǔn)庫中的 allocator 是多余的  回復(fù)  更多評(píng)論   

這種材料的主題是非常有趣的,令人驚異的是看到了明顯的交談here.Keep了出色的執(zhí)行,這要感謝提出建議,這些信息
2014-01-09 11:58 | essays
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲欧美成人| 欧美一区二区三区在线观看| 亚洲美女av黄| 亚洲激情小视频| 18成人免费观看视频| 国精品一区二区三区| 国产午夜精品全部视频在线播放| 国产精品久久久一区二区| 国产精品高潮呻吟久久| 国产性做久久久久久| 亚洲国产高清高潮精品美女| 亚洲精品久久久久久久久久久久 | 国产日韩欧美综合| 精品成人在线| 一本大道av伊人久久综合| 性高湖久久久久久久久| 久久久久se| 欧美肥婆在线| 一区二区三区久久精品| 亚洲自拍三区| 欧美大片在线观看一区| 欧美激情中文字幕一区二区| 国产精品美女久久久久aⅴ国产馆| 国产精品久久77777| 精品动漫3d一区二区三区免费| 在线国产精品播放| 亚洲午夜精品久久久久久浪潮| 欧美一区日韩一区| 欧美激情一区| 香蕉成人久久| 欧美精品久久久久久久久久| 国产三区精品| 中文精品视频| 欧美jizzhd精品欧美巨大免费| 在线一区视频| 欧美精品一区二区高清在线观看| 国产精品你懂的在线| 亚洲人午夜精品| 久久亚洲免费| 欧美在线黄色| 国产欧美亚洲精品| 中文精品视频| 日韩视频免费在线| 嫩草影视亚洲| 在线观看欧美日本| 欧美在线播放| 亚洲一区二区三区影院| 欧美日韩三级视频| 亚洲美女在线看| 免费看精品久久片| 久久久亚洲午夜电影| 国模精品娜娜一二三区| 性欧美暴力猛交另类hd| 亚洲视频一起| 国产精品日本欧美一区二区三区| 99re6热在线精品视频播放速度| 欧美国产精品人人做人人爱| 亚欧成人在线| 国产综合欧美在线看| 欧美在线视频一区二区三区| 亚洲欧美日韩在线观看a三区| 欧美四级在线观看| 日韩亚洲欧美一区| 久久人人97超碰人人澡爱香蕉| 亚洲女人天堂成人av在线| 欧美亚一区二区| 欧美亚洲一级| 亚洲一区二区三区视频| 国产日韩欧美一区二区| 久久经典综合| 狂野欧美激情性xxxx| 亚洲福利视频一区二区| 欧美激情精品久久久久久大尺度| 久久久久久久波多野高潮日日| 伊人狠狠色丁香综合尤物| 欧美成人免费在线观看| 久久久久久久性| 亚洲精一区二区三区| 亚洲欧洲一二三| 欧美国产亚洲另类动漫| 亚洲视频一区在线观看| 性视频1819p久久| 影音先锋久久资源网| 日韩亚洲视频| 亚洲一区二区三区三| 在线一区二区三区四区| 欧美日韩国产不卡| 亚洲影视综合| 久久精品一区二区三区中文字幕| 亚洲国产经典视频| 99riav久久精品riav| 国产欧美日韩一区二区三区| 久久这里只精品最新地址| 欧美国产亚洲另类动漫| 午夜精品福利视频| 久久精品一区二区三区不卡牛牛 | 国产精品久久九九| 久久人人超碰| 欧美日韩国产探花| 久久国产精品久久久| 欧美黑人在线观看| 久久久久久久久伊人| 欧美男人的天堂| 久久精品夜色噜噜亚洲a∨| 欧美精品 国产精品| 久久精品国产99| 欧美美女bb生活片| 欧美jizz19性欧美| 国产一区久久| 一区二区三区四区国产| 国产日韩视频| 一区二区三区不卡视频在线观看| 激情五月***国产精品| 一区二区欧美日韩| 亚洲精品一区二区三| 久久国产精品亚洲77777| 亚洲社区在线观看| 欧美aaaaaaaa牛牛影院| 免费影视亚洲| 黄色成人在线| 午夜一区二区三区不卡视频| 中文国产一区| 欧美日韩国产天堂| 亚洲黄色有码视频| 在线观看国产精品淫| 久久成人免费视频| 欧美一区二区黄| 国产精品久在线观看| 99在线热播精品免费| 日韩一区二区高清| 欧美精品日韩一区| 亚洲黄色免费电影| 91久久久久久久久| 欧美成人免费在线视频| 久久人人97超碰国产公开结果| 国产日韩一区二区三区| 午夜精品久久久久久久白皮肤| 欧美一级视频| 国户精品久久久久久久久久久不卡| 亚洲自啪免费| 欧美一区二区三区在线视频| 国产精品天美传媒入口| 亚洲一级免费视频| 午夜精品电影| 国产九色精品成人porny| 国产一区二区成人| 亚洲欧美国产视频| 欧美日韩亚洲一区二区三区在线观看| 亚洲激情影视| 中文欧美在线视频| 欧美午夜寂寞影院| 黄色在线成人| 久久久久久久波多野高潮日日| 欧美成人一区二免费视频软件| 亚洲电影视频在线| 欧美福利网址| 亚洲色图在线视频| 久久久久久久波多野高潮日日| 在线看国产日韩| 欧美激情一区二区三区| 一区二区高清视频在线观看| 久久精品91久久香蕉加勒比| 激情av一区二区| 欧美成人精品福利| 亚洲在线观看视频网站| 美女主播一区| av不卡在线看| 国产精品视频一| 久久综合免费视频影院| 亚洲三级电影全部在线观看高清| 亚洲精品小视频在线观看| 欧美交受高潮1| 亚洲一区黄色| 亚洲国产女人aaa毛片在线| 亚洲天堂av综合网| 狠狠综合久久av一区二区小说| 欧美—级a级欧美特级ar全黄| 亚洲午夜女主播在线直播| 玖玖综合伊人| 亚洲美女av在线播放| 国产亚洲综合在线| 欧美成人午夜激情在线| 亚洲欧美日韩一区在线观看| 亚洲国产视频a| 久久在线免费观看| 欧美在线视频网站| 亚洲视频第一页| 亚洲二区在线视频| 夜久久久久久| 欧美成人午夜激情| 久久米奇亚洲| 欧美一区二区三区喷汁尤物| 99视频有精品| 亚洲欧洲一区二区在线播放| 国产婷婷色一区二区三区| 国产精品豆花视频| 欧美日韩国产美女| 麻豆91精品| 久久亚裔精品欧美| 久久福利精品|