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

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

讓我們先來看看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;
由上可見,正如上一篇所說,vector的迭代器是由原生的指針來實現的。

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

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

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

下面來看看vector中最基本的構造函數和析構函數
            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個成員變量都填充為0。
上一篇所說,在STL中是將內存分配與對象初始化分開的,同樣對象析構與內存釋放也是被分開的。

然后是其begin和end方法,用來獲取第一個元素和最后一個元素的后一個元素的迭代器。
            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分別被用來獲取第一個元素和最后一個元素
            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分別被用來判別容器是否為空、容器內元素的個數和容器的大小
            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的頭部插入元素會使所有元素后移,應此它被設計為單向的,只能由尾部插入或移除數據
            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));
            }
當然從頭部移除數據也并非不可以,可以使用erase方法來移除頭部的數據,erase方法將會在后面的部分作出說明。

我們先來看一下insert_aux這個方法,在插入元素時幾乎都使用到了這個方法。
            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;
                }
            }
在容器還有足夠的空間時,首先將從position位置到finish位置的元素整體后移一個位置,最后將要被插入的元素寫入到原position的位置同時改變finish指針的值。
若空間不足時,首先根據原有空間的大小的一倍來申請內存,然后將元素從原有位置的begin到position拷貝到新申請的內存中,然后在新申請內存的指定位置插入要插入的元素值,最后將余下的部分也拷貝過來。然后將原有元素析構掉并把內存釋放掉。

為何不使用reallocate?
reallocate的本意并不是在原有內存的位置增加或減少內存,reallocate首先會試圖在原有的內存位置增加或減少內存,若失敗則會重新申請一塊新的內存并把原有的數據拷貝過去,這種操作本質上等價于重新申請一塊內存,應此這里使用的是allocate而并非reallocate。

然后讓我們來看一下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;
            }
若是要在最后插入一個元素且容器的剩余空間還足夠的話,直接將元素插入到finish的位置,并將finish指針后移一位即可。若容器空間不夠或不是插在最后一個的位置,則調用insert_aux重新分配內存或插入。
刪除時首先析構掉原有元素,若被刪元素不是最后一個元素,則將后面的所有元素拷貝過來,最后將finish指針前移一個位置。

最后讓我們來看一下其中重載的運算符
            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內部用的是原生的指針,應此這些運算符的使用方式和原生指針的并無差異。值得注意的是在做賦值操作時會產生內存的重新分配與拷貝操作。

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

評論:
# re: 山寨STL實現之vector[未登錄] 2012-06-18 11:22 | 路人甲
“迭代器”比較麻煩,得做到自己版本的vector的迭代器能拿到std中使用。
  回復  更多評論
  
# re: 山寨STL實現之vector 2012-06-26 00:58 | 朱峰everettjf
學習,
自己最近看了STL源碼剖析,總是在看,卻未想自己實現一個。
  回復  更多評論
  
# re: 山寨STL實現之vector 2012-06-26 09:42 | lwch
@朱峰everettjf
程序還是要靠多寫才行~  回復  更多評論
  
# re: 山寨STL實現之vector 2012-06-30 21:28 | beibei
博主,我剛好也在做這個,但在實現含有兩個迭代器的構造函數時,出了問題:
template<class In>
Vec(In b, In e) {
start=alloc.allocate( e-b );
finish=end_of_element=std::uninitialized_copy( b, e, data );
}
在最后一行出現了error C2665。  回復  更多評論
  
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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ⅴ9色| 亚洲亚洲精品在线观看| 亚洲午夜在线观看视频在线| 国产偷自视频区视频一区二区| 久久久久.com| 欧美激情一区二区三区| 亚洲欧美日韩专区| 久久精品日产第一区二区三区| 亚洲精品激情| 午夜精品久久久久久久99水蜜桃 | 欧美国产日韩a欧美在线观看| 亚洲麻豆av| 午夜精品久久久久久| 亚洲国产成人精品女人久久久| 日韩西西人体444www| 国产亚洲福利一区| 亚洲欧洲综合另类| 国产精品丝袜xxxxxxx| 免费不卡在线观看av| 欧美日韩一区二区精品| 久久亚洲欧洲| 国产精品色婷婷| 亚洲国产成人porn| 国产视频精品免费播放| 亚洲国产你懂的| 国外视频精品毛片| 亚洲视频第一页| 亚洲另类在线一区| 久久精品一二三区| 性欧美办公室18xxxxhd| 欧美黄色免费网站| 免费短视频成人日韩| 国产日本欧美一区二区三区| 亚洲人屁股眼子交8| 一区福利视频| 亚洲在线观看视频| 亚洲色诱最新| 欧美精品一区二区三区在线播放| 久久精品国产亚洲一区二区三区| 欧美日韩国产二区| 亚洲国产精品电影| 亚洲电影欧美电影有声小说| 性欧美在线看片a免费观看| 亚洲午夜国产成人av电影男同| 久久尤物视频| 欧美 日韩 国产一区二区在线视频 | 亚洲欧美日韩人成在线播放| 中文在线资源观看网站视频免费不卡| 另类亚洲自拍| 猫咪成人在线观看| 国内精品福利| 欧美一级欧美一级在线播放| 翔田千里一区二区| 欧美亚洲第一区| 亚洲手机成人高清视频| 一本色道婷婷久久欧美| 欧美人牲a欧美精品| 亚洲精品女av网站| 一区在线电影| 麻豆精品精华液| 亚洲大片精品永久免费| 亚洲欧洲在线一区| 欧美精品在线免费观看| 亚洲人成亚洲人成在线观看| 亚洲精品一区二区三区樱花| 欧美高清不卡| 一区二区不卡在线视频 午夜欧美不卡在| 日韩亚洲国产精品| 国产精品99免视看9| 亚洲欧美在线aaa| 久久久久久亚洲精品杨幂换脸| 精品动漫3d一区二区三区免费版| 久久久精彩视频| 欧美国产丝袜视频| 亚洲午夜国产一区99re久久| 国产精品区一区二区三| 久久激情视频| 亚洲国产专区校园欧美| 亚洲伊人久久综合| 狠狠久久综合婷婷不卡| 免费成人高清| 一区二区三区四区五区在线| 久久精品中文字幕免费mv| 亚洲国产高清在线| 欧美视频在线观看视频极品| 欧美一级专区| 91久久综合| 久久精品人人| 亚洲三级免费| 国产亚洲一区二区三区在线播放| 久久中文字幕一区二区三区| 日韩午夜在线观看视频| 久久久久久久久伊人| 9人人澡人人爽人人精品| 国产伦一区二区三区色一情| 麻豆乱码国产一区二区三区| 亚洲婷婷国产精品电影人久久| 久久只有精品| 亚洲自拍啪啪| 亚洲精品一区二区在线观看| 国产嫩草影院久久久久 | 亚洲一区免费| 亚洲国产精品一区| 欧美制服第一页| 亚洲免费观看| 精品福利免费观看| 国产精品国产三级国产专区53| 久久亚洲精品视频| 香蕉免费一区二区三区在线观看| 亚洲片在线观看| 欧美成人亚洲成人日韩成人| 久久www成人_看片免费不卡| 一本一本a久久| 91久久精品国产91久久性色tv| 国产欧美日韩视频一区二区三区| 欧美精品v日韩精品v国产精品| 久久久久国产一区二区| 亚洲欧美乱综合| 中文国产一区| 日韩视频专区| 日韩视频三区| 亚洲黄色尤物视频| 欧美激情成人在线| 美女啪啪无遮挡免费久久网站| 欧美中文在线观看| 性8sex亚洲区入口| 亚洲欧美另类在线观看| 亚洲私人黄色宅男| 亚洲午夜精品久久久久久浪潮| 亚洲精选久久| 99国产欧美久久久精品| 99精品欧美一区二区三区综合在线 | 国产精品爱久久久久久久| 欧美福利视频在线| 欧美精品91| 欧美日韩一区三区| 国产精品成人午夜| 国产精品国码视频| 国产精品日韩精品欧美精品| 国产精品国产a级| 国产精品亚发布| 国产一区二区三区电影在线观看| 国产日韩欧美中文在线播放| 国产区欧美区日韩区| 国产欧美一区在线| 激情婷婷欧美| 亚洲美女毛片| 亚洲影院一区| 久久久久国产精品一区二区| 久久亚洲精选| 亚洲国产高清视频| 一区二区三区色| 欧美在线视频观看免费网站| 久久久免费av| 欧美乱大交xxxxx| 国产精品国产三级国产aⅴ浪潮| 国产精品日韩在线观看| 精品51国产黑色丝袜高跟鞋| 最近看过的日韩成人| 亚洲一区免费看| 久久亚洲精品中文字幕冲田杏梨| 欧美激情在线有限公司| 一区二区三区欧美| 久久久久国产一区二区| 欧美激情第六页| 国产老肥熟一区二区三区| 尤物九九久久国产精品的分类| 亚洲日本中文字幕免费在线不卡| 亚洲欧美国产va在线影院| 久久精品人人做人人爽| 欧美激情网友自拍| 亚洲夜晚福利在线观看| 可以免费看不卡的av网站| 欧美视频在线播放| 亚洲电影下载| 香蕉成人久久| 亚洲黄色成人网| 欧美在线|欧美| 欧美午夜精彩| 伊人伊人伊人久久| 亚洲免费视频成人| 欧美黄色免费| 久久爱www.| 国产精品v欧美精品v日韩| 亚洲国产精品va| 久久精品在线播放| 一区二区精品| 欧美岛国在线观看| 在线播放中文一区| 久久色中文字幕| 亚洲一区二区三区在线播放| 欧美黑人一区二区三区| 在线不卡视频|