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

隨筆-91  評(píng)論-137  文章-0  trackbacks-0
首先是vector的定義
        template <typename T>
        class vector
        {
        };

讓我們先來(lái)看看vector中的一些別名
        public:
            typedef T         value_type;
            typedef T*        pointer;
            typedef T&        reference;
            typedef const T&  const_reference;
            typedef size_t    size_type;
            typedef ptrdiff_t difference_type;
            typedef const T* const_iterator;
            typedef reverse_iterator<const_iterator, value_type, size_type, difference_type> const_reverse_iterator;
            typedef T* iterator;
            typedef reverse_iterator<iterator, value_type, size_type, difference_type> reverse_iterator;
由上可見(jiàn),正如上一篇所說(shuō),vector的迭代器是由原生的指針來(lái)實(shí)現(xiàn)的。

下面是其內(nèi)部的成員變量
        protected:
            typedef vector<T>    self;
            typedef allocator<T> Alloc;

            iterator start;
            iterator finish;
            iterator end_of_element;
start:指向vector的起始地址
finish:指向最后一個(gè)元素的后一個(gè)元素的地址
end_of_element:指向已申請(qǐng)內(nèi)存塊的結(jié)束位置(finish總是小于或等于end_of_element)

在STL中從begin到end總是以一個(gè)前閉后開(kāi)的形式來(lái)表示的,形如[begin,end),這樣做的好處是可以使代碼寫(xiě)的更簡(jiǎn)潔:
        template <typename Iterator, typename T>
        Iterator find(Iterator first, Iterator last, const T& value)
        {
            while(first != last && *first != value) ++first;
            return first;
        }

下面來(lái)看看vector中最基本的構(gòu)造函數(shù)和析構(gòu)函數(shù)
            vector() : start(0), finish(0), end_of_element(0)
            {
            }

            ~vector()
            {
                destruct(start, end_of_element);
                if (start != 0) Alloc::deallocate(start, end_of_element - start);
            }
這里將其中的3個(gè)成員變量都填充為0。
上一篇所說(shuō),在STL中是將內(nèi)存分配與對(duì)象初始化分開(kāi)的,同樣對(duì)象析構(gòu)與內(nèi)存釋放也是被分開(kāi)的。

然后是其begin和end方法,用來(lái)獲取第一個(gè)元素和最后一個(gè)元素的后一個(gè)元素的迭代器。
            inline iterator begin()
            {
                return start;
            }

            inline const_iterator begin()const
            {
                return start;
            }

            inline iterator end()
            {
                return finish;
            }

            inline const_iterator end()const
            {
                return finish;
            }

front和back分別被用來(lái)獲取第一個(gè)元素和最后一個(gè)元素
            inline reference front()
            {
                return *begin();
            }

            inline const_reference front()const
            {
                return *begin();
            }

            inline reference back()
            {
                return *(end() - 1);
            }

            inline const_reference back()const
            {
                return *(end() - 1);
            }

empty、size、capacity分別被用來(lái)判別容器是否為空、容器內(nèi)元素的個(gè)數(shù)和容器的大小
            inline bool empty()const
            {
                return begin() == end();
            }

            inline const size_type size()const
            {
                return size_type(end() - begin());
            }

            inline const size_type capacity()const
            {
                return size_type(end_of_element - begin());
            }

由于在vector的頭部插入元素會(huì)使所有元素后移,應(yīng)此它被設(shè)計(jì)為單向的,只能由尾部插入或移除數(shù)據(jù)
            void push_back(const T& x)
            {
                if (end_of_element != finish)
                {
                    construct(&*finish, x);
                    ++finish;
                }
                else
                {
                    insert_aux(end(), x);
                }
            }

            inline void pop_back()
            {
                --finish;
                destruct<T>(finish, has_destruct(*finish));
            }
當(dāng)然從頭部移除數(shù)據(jù)也并非不可以,可以使用erase方法來(lái)移除頭部的數(shù)據(jù),erase方法將會(huì)在后面的部分作出說(shuō)明。

我們先來(lái)看一下insert_aux這個(gè)方法,在插入元素時(shí)幾乎都使用到了這個(gè)方法。
            void insert_aux(const iterator position, const T& x)
            {
                if(finish != end_of_element)
                {
                    construct(&*finish, *(finish - 1));
                    T x_copy = x;
                    copy_backward(position, finish - 1, finish);
                    *position = x_copy;
                    ++finish;
                }
                else
                {
                    const size_type old_size = size();
                    const size_type new_size = old_size == 0 ? 2 : old_size * 2;
                    iterator tmp = Alloc::allocate(new_size);
                    uninitialized_copy(begin(), position, tmp);
                    iterator new_position = tmp + (position - begin());
                    construct(&*new_position, x);
                    uninitialized_copy(position, end(), new_position + 1);
                    destruct(begin(), end());
                    Alloc::deallocate(begin(), old_size);
                    end_of_element = tmp + new_size;
                    finish = tmp + old_size + 1;
                    start = tmp;
                }
            }
在容器還有足夠的空間時(shí),首先將從position位置到finish位置的元素整體后移一個(gè)位置,最后將要被插入的元素寫(xiě)入到原position的位置同時(shí)改變finish指針的值。
若空間不足時(shí),首先根據(jù)原有空間的大小的一倍來(lái)申請(qǐng)內(nèi)存,然后將元素從原有位置的begin到position拷貝到新申請(qǐng)的內(nèi)存中,然后在新申請(qǐng)內(nèi)存的指定位置插入要插入的元素值,最后將余下的部分也拷貝過(guò)來(lái)。然后將原有元素析構(gòu)掉并把內(nèi)存釋放掉。

為何不使用reallocate?
reallocate的本意并不是在原有內(nèi)存的位置增加或減少內(nèi)存,reallocate首先會(huì)試圖在原有的內(nèi)存位置增加或減少內(nèi)存,若失敗則會(huì)重新申請(qǐng)一塊新的內(nèi)存并把原有的數(shù)據(jù)拷貝過(guò)去,這種操作本質(zhì)上等價(jià)于重新申請(qǐng)一塊內(nèi)存,應(yīng)此這里使用的是allocate而并非reallocate。

然后讓我們來(lái)看一下insert和erase方法
            inline iterator insert(iterator position, const T& x)
            {
                const size_type pos = position - begin();
                if(finish != end_of_element && position == end())
                {
                    construct(&*finish, x);
                    ++finish;
                }
                else insert_aux(position, x);
                return begin() + pos;
            }

            iterator erase(iterator position)
            {
                destruct(position, has_destruct(*position));
                if (position + 1 != end())
                {
                    copy(position + 1, end(), position);
                }
                --finish;
                return position;
            }
若是要在最后插入一個(gè)元素且容器的剩余空間還足夠的話,直接將元素插入到finish的位置,并將finish指針后移一位即可。若容器空間不夠或不是插在最后一個(gè)的位置,則調(diào)用insert_aux重新分配內(nèi)存或插入。
刪除時(shí)首先析構(gòu)掉原有元素,若被刪元素不是最后一個(gè)元素,則將后面的所有元素拷貝過(guò)來(lái),最后將finish指針前移一個(gè)位置。

最后讓我們來(lái)看一下其中重載的運(yùn)算符
            self& operator=(const self& x)
            {
                if(&x == thisreturn *this;
                size_type const other_size = x.size();
                if(other_size > capacity())
                {
                    destruct(start, finish);
                    Alloc::deallocate(start, capacity());
                    start = Alloc::allocate(other_size);
                    finish = uninitialized_copy(x.begin(), x.end(), start);
                    end_of_element = start + other_size;
                }
                else
                {
                    finish = uninitialized_copy(x.begin(), x.end(), start);
                }
                return *this;
            }

            inline reference operator[](size_type n)
            {
                return *(begin() + n);
            }

            inline value_type at(size_type n)
            {
                return *(begin() + n);
            }
由于vector內(nèi)部用的是原生的指針,應(yīng)此這些運(yùn)算符的使用方式和原生指針的并無(wú)差異。值得注意的是在做賦值操作時(shí)會(huì)產(chǎn)生內(nèi)存的重新分配與拷貝操作。

至此,vector的講解已完成,完整的代碼請(qǐng)到http://qlanguage.codeplex.com下載
posted on 2012-06-17 17:08 lwch 閱讀(3608) 評(píng)論(3)  編輯 收藏 引用 所屬分類(lèi): STL

評(píng)論:
# re: 山寨STL實(shí)現(xiàn)之vector[未登錄](méi) 2012-06-18 11:22 | 路人甲
“迭代器”比較麻煩,得做到自己版本的vector的迭代器能拿到std中使用。
  回復(fù)  更多評(píng)論
  
# re: 山寨STL實(shí)現(xiàn)之vector 2012-06-26 00:58 | 朱峰e(cuò)verettjf
學(xué)習(xí),
自己最近看了STL源碼剖析,總是在看,卻未想自己實(shí)現(xiàn)一個(gè)。
  回復(fù)  更多評(píng)論
  
# re: 山寨STL實(shí)現(xiàn)之vector 2012-06-26 09:42 | lwch
@朱峰e(cuò)verettjf
程序還是要靠多寫(xiě)才行~  回復(fù)  更多評(píng)論
  
# re: 山寨STL實(shí)現(xiàn)之vector 2012-06-30 21:28 | beibei
博主,我剛好也在做這個(gè),但在實(shí)現(xiàn)含有兩個(gè)迭代器的構(gòu)造函數(shù)時(shí),出了問(wèn)題:
template<class In>
Vec(In b, In e) {
start=alloc.allocate( e-b );
finish=end_of_element=std::uninitialized_copy( b, e, data );
}
在最后一行出現(xiàn)了error C2665。  回復(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>
            久久中文久久字幕| 日韩一区二区免费高清| 影音先锋成人资源站| 国产日韩精品视频一区| 国产女人精品视频| 国产一区二区看久久| 激情欧美一区二区三区| 91久久国产综合久久91精品网站| 激情综合自拍| 亚洲人在线视频| 亚洲无吗在线| 久久国产精品毛片| 欧美福利电影网| 日韩一级黄色片| 久久国产手机看片| 欧美理论电影在线观看| 国产精品盗摄一区二区三区| 国产色综合网| 亚洲精选久久| 久久天天躁狠狠躁夜夜爽蜜月| 欧美www在线| 亚洲五月婷婷| 欧美成人国产va精品日本一级| 欧美三区在线视频| 激情懂色av一区av二区av| 日韩午夜在线电影| 久久久久久久一区二区三区| 一区二区三区黄色| 欧美大尺度在线| 亚洲永久在线观看| 蜜桃久久精品乱码一区二区| 亚洲人成小说网站色在线| 亚洲日本欧美天堂| 久久av红桃一区二区小说| 欧美日韩国产综合网 | 欧美专区在线| 欧美日韩精品二区| 亚洲成色777777在线观看影院| 亚洲视屏在线播放| 欧美激情中文不卡| 欧美专区福利在线| 国产精品欧美在线| 一本色道**综合亚洲精品蜜桃冫| 久久精品99国产精品日本| 99国产精品| 欧美精品一区二区久久婷婷| 精久久久久久久久久久| 欧美影院成年免费版| 日韩一级免费| 亚洲高清视频一区| 日韩系列在线| 欧美精彩视频一区二区三区| 影音先锋久久资源网| 久久国产精品黑丝| 亚洲欧美中日韩| 国产精品欧美一区喷水 | 欧美日韩成人免费| 亚洲三级毛片| 亚洲第一精品电影| 久色婷婷小香蕉久久| 好吊色欧美一区二区三区四区| 午夜久久美女| 亚洲欧美日韩一区二区三区在线观看| 欧美日韩在线不卡| 亚洲视频免费观看| 中文久久精品| 国产啪精品视频| 久久久久久久波多野高潮日日| 小处雏高清一区二区三区 | 亚洲毛片在线观看.| 欧美大片在线观看一区| 久久久一二三| 日韩亚洲精品在线| 99在线视频精品| 国产精品视频一区二区三区| 性做久久久久久久免费看| 一区二区三区精品视频在线观看| 欧美三级在线播放| 欧美一区视频| 久热精品在线视频| 在线视频免费在线观看一区二区| 在线视频免费在线观看一区二区| 国产精品免费看片| 久久精品亚洲精品| 美女黄色成人网| 一区二区欧美国产| 欧美一区二区网站| 亚洲欧洲精品天堂一级| 亚洲三级性片| 国产精品久久77777| 久久国产欧美日韩精品| 久久在线免费| 亚洲一区二区三区视频播放| 性欧美xxxx大乳国产app| 亚洲国产黄色片| 日韩午夜电影av| 国产亚洲激情| 日韩午夜免费视频| 一区二区三区在线免费视频 | 嫩草影视亚洲| 国产精品v片在线观看不卡| 欧美一区二区三区视频免费| 久久久久久久久岛国免费| 一区二区三区欧美成人| 久久精品亚洲乱码伦伦中文| 亚洲免费av片| 午夜久久资源| 欧美日韩久久| 久久久免费av| 国产精品jizz在线观看美国| 久久视频一区二区| 国产精品久久久久久久久| 欧美激情精品久久久久久免费印度 | 狠狠色狠狠色综合| 亚洲免费观看| 亚洲国语精品自产拍在线观看| 亚洲一区二区三区中文字幕在线 | 欧美伊人久久久久久久久影院| 久久久久综合网| 欧美一区二区视频在线| 欧美日韩情趣电影| 亚洲高清一区二| 永久免费精品影视网站| 亚洲欧美日韩国产| 亚洲永久免费精品| 欧美日韩黄视频| 日韩视频免费大全中文字幕| 91久久综合| 欧美电影免费观看网站 | 亚洲一区二区日本| 一本色道久久88精品综合| 蜜臀99久久精品久久久久久软件 | 国产精品国产亚洲精品看不卡15| 亚洲国产激情| 亚洲精品影院在线观看| 欧美高清在线视频| 亚洲精品乱码久久久久| 亚洲免费观看高清在线观看| 欧美+亚洲+精品+三区| 欧美va亚洲va日韩∨a综合色| 尤物精品国产第一福利三区| 久久久爽爽爽美女图片| 狂野欧美激情性xxxx| 在线观看av不卡| 欧美xx69| 夜夜嗨av一区二区三区四季av| 在线午夜精品自拍| 国产精品国产自产拍高清av王其| 亚洲一区影音先锋| 欧美一区永久视频免费观看| 国产一区二区三区久久精品| 久久国产精品亚洲77777| 久久视频在线视频| 亚洲三级视频| 国产精品久久久久免费a∨大胸| 亚洲色在线视频| 久久久久久久久久久久久女国产乱| 精品av久久707| 欧美激情视频一区二区三区在线播放| 亚洲精品系列| 久久av一区二区三区漫画| 一区二区三区我不卡| 欧美精品播放| 亚洲欧美文学| 欧美成人午夜免费视在线看片| 亚洲精品在线观看视频| 国产精品都在这里| 久久黄金**| 亚洲精品在线免费| 欧美国产欧美亚洲国产日韩mv天天看完整 | 午夜久久福利| 在线不卡a资源高清| 欧美女激情福利| 午夜精品一区二区三区电影天堂| 久久综合久色欧美综合狠狠| 日韩视频一区二区三区在线播放| 国产精品夜夜嗨| 欧美成人四级电影| 午夜欧美精品久久久久久久| 欧美成人免费网站| 午夜天堂精品久久久久| 亚洲国产精品va在线观看黑人| 欧美日韩综合| 免费短视频成人日韩| 亚洲综合成人婷婷小说| 欧美激情视频一区二区三区在线播放 | 欧美丝袜第一区| 久久久久久久精| 亚洲永久免费av| 亚洲人成人99网站| 免费欧美电影| 久久九九精品| 午夜在线成人av| 在线视频欧美一区| 亚洲人人精品| 亚洲国产日韩欧美在线动漫| 国产精品视频男人的天堂| 欧美日韩在线看| 欧美日韩国内| 欧美日韩高清不卡|