日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

STL源码剖析 deque双端队列 概述

發布時間:2023/12/13 编程问答 45 豆豆
生活随笔 收集整理的這篇文章主要介紹了 STL源码剖析 deque双端队列 概述 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
  • vector是單向開口的連續線性空間,deque是一種雙向開口的連續線性空間。
  • deque可以在頭尾兩端分別進行元素的插入和刪除操作

  • ?vector和deque的差異
    • 1,deque允許常數時間內對于頭端元素進行插入和刪除操作
    • 2,deque沒有所謂容量(capacity)的概念,因為它是動態的以分段連續空間組合而成的,隨時可以增加一段空間,將其銜接起來。deque不會發生vector那樣的“因舊的空間不足而重新配置一段內存區間,然后進行元素的復制和舊的空間的釋放”,因此deque不具備所謂的空間保留的能力,也就是reverse
  • 雖然deque也提供Random Access Iterator,但是他的迭代器不是普通的指針,其復雜度較高,如果可以使用vector就不要使用deque
  • 對于deque進行的排序操作,為了較高的效率,可以將deque先完整復制到一個vector內部,進行排序(STL sort算法) 再復制回deque
  • deque由一段一段的定量連續的空間構成。如果需要擴展空間,需要先配置一段定量的連續空間,將其串聯在deque的頭部或者尾部。deque的最大的任務就是在這些分段的定量的連續空間上,維護其整體連續的假象,并提供隨機存取的接口,相較于vector只能向尾端成長的假象,實際需要進行“重新配置、復制、釋放”的輪回操作而言,代價是需要設計 復雜的迭代器架構
  • deque使用分段連續性空間,是需要一個中央控制器進行統一調控,但是為了維持整體連續的假象,數據結構的設計和迭代器的前進和后退操作等頗為繁瑣
  • deque采用一塊所謂的map (這里不是指STL map容器)作為主控,這里的map是指一段連續的空間,其中每個元素(此處稱為一個節點,node)都是指針,指向的是一段(較大的)連續內存空間,稱其為緩沖區。
  • deque的存儲空間的主體是 緩沖區
  • SGI STL允許我們指定緩沖區的大小,默認值0表示使用512bytes緩沖區
  • map本質是 T** ,即指針的指針,指向的是型別為T的一塊空間

?deque迭代器

  • deque是分段連續的空間。需要維持其整體連續的假象的任務是通過operator++和operator--兩個運算子實現的
  • 迭代器應該具備什么結構呢?從功能點分析:1,指出分段連續空間(緩沖區)在哪里,從而判斷自己是否已經處于緩沖區的邊界處,一旦前進或者后退就必須跳躍至下一個或者上一個緩沖區。為了可以正確跳躍,需要隨之掌控 控制中心(map)

  • ?假設產生一個deque<int>令其緩沖區的大小為32,那么每個緩沖區可以容納的元素個數是 32/sizeof(int) =8個元素。假設deque存儲20個元素,需要20/8=3個緩沖區,所以map中控節點需要使用3個節點。
  • 迭代器start內的cur指針當然指向緩沖區的第一個元素,迭代器finish內的cur指針指向的是第三個緩沖區的最后元素(的下一個位置)??紤]到最后一個緩沖區尚有備用的空間,如果有新的元素插入到尾端,可以直接使用這個備用空間。
  • 陰影區域表示? 還存在四個區域尚未使用

  • 對于各個指針的運算比如加、減、前進、后退都不可以直觀看到。
  • 最關鍵的是:一旦行進的時候遇到緩沖區邊緣,需要特別當心,需要視前進后退而定,可能需要使用set_node函數跳一個緩沖區
  • 雙端隊列的迭代器的代碼
/** 如果n不為0,傳回n,表示buffer size由用戶自定義* 如果n為0,表示buffer size使用的是默認數值,那么* 如果sz(元素的大小,sizeof(value_type))小于512,傳回 512/sz* 如果sz不小于512,傳回1*/ inline std::size_t __deque_buf_size(std::size_t n,std::size_t sz){return n!=0 ? n : (sz < 512 ? std::size_t (512/sz):std::size_t(1)); }//雙端隊列 迭代器 template<class T,class Ref,class Ptr,std::size_t BufSiz> struct __deque_iterator{ //未繼承 std::iteratortypedef __deque_iterator<T,T&,T*,BufSiz> iterator;typedef __deque_iterator<T,const T&,const T*,BufSiz> const_iterator;static std::size_t buffer_size() {return __deque_buf_size(BufSiz,sizeof (T));}//未繼承 std::iterator,所以需要自行撰寫五個必要的迭代器對應的類別typedef std::random_access_iterator_tag iterator_category;//1typedef T value_type; //2typedef Ptr pointer; //3typedef Ref reference; //4typedef std::size_t size_type;typedef ptrdiff_t difference_type; //5typedef T** map_pointer;typedef __deque_iterator self;//保持和容器的聯結T* cur; //此迭代器所指向的緩沖區的現行(current)元素T* first; //此迭代器所指之緩沖區的頭T* last; //此迭代器所指之緩沖區的尾 (含備用空間)map_pointer node;//指向管控中心public:void set_node(map_pointer new_node){node = new_node;first = *new_node;last = first + difference_type (buffer_size());}//重載各個運算子 __deque_iterator<> 成功運作的關鍵reference operator*() const {return *cur;}pointer operator->() const {return &(operator*());}difference_type operator-(const self& x) const{return difference_type (buffer_size()) * (node - x.node - 1) +(cur - first) + (x.last - x.cur);}//參考More Effective C++//Distinguish between prefix and postfix of increment and decrement operators//注意last指向的是最后一個節點的后面,所以先遞增再判斷self& operator++(){++cur; //切換至下一個元素if (cur==last){ //如果已經到達所在緩沖區的尾端set_node(node+1);//切換至下一個節點 (亦 即緩沖區的第一個元素)cur = first;}return *this;}self& operator++(int){ //后置式 標準寫法self tmp = *this;++*this; //調用self& operator++()return tmp;}//first指向的是最后一個節點,所以需要先進行判斷self& operator--(){if (cur == first){ //如果已經到達所在緩沖區的頭端set_node(node-1);//切換至前一個節點(亦即緩沖區)的最后一個元素cur = last;}--cur; //切換至前一個元素return *this;}self& operator--(int){ //后置式 標準寫法self tmp = *this;--*this;return tmp;}//以下代碼實現隨機存取,迭代器直接跳過 n 個距離self& operator+=(difference_type n){difference_type offset = n + (cur - first);if (offset >= 0 && offset < difference_type(buffer_size())){//目標位置在同一緩沖區內cur += n;} else{//目標位置不在同一個緩沖區內difference_type node_offset = offset > 0 ? offset / difference_type(buffer_size()): -difference_type ((-offset - 1) / buffer_size()) - 1;//切換到正確的節點(即對應的緩沖區)set_node(node+node_offset);//切換至正確的元素cur = first+(offset - node_offset * difference_type(buffer_size()));}return *this;}//參考More Effective C++ item22//consider using op= instead of stand-alone opself operator+(difference_type n) const{self tmp = *this;return tmp += n; //調用operator+=}//通過使用operator+= 完成operator-=self& operator-=(difference_type n){return *this += -n;}self operator-(difference_type n)const{self tmp = *this;return tmp -= n;//調用operator-=}//以下實現隨機存取,迭代器可以直接跳躍n個距離reference operator[](difference_type n)const {//以上調用operator* ,operator+return *(*this + n);}bool operator==(const self& x)const{ return cur == x.cur;}bool operator!=(const self& x)const{ return cur != x.cur;}bool operator< (const self& x)const{return (node == x.node) ? (cur < x.cur) : (node < x.node);} };

deque的數據結構

  • deque除了需要維護一個先前說過的指向map的指針外,也需要維護start和finish兩個迭代器,分別指向第一緩沖區的第一個元素和最后緩沖區的最后一個元素(的下一個位置)
  • 注意:還需要記住此刻的map的大小,因為一旦map所提供的節點不足,就需要重新配置更大的一塊map
//雙端隊列 template<class T,class Alloc,std::size_t BufSiz = 0> class deque{ public: //Basic typestypedef T value_type;typedef value_type* pointer;typedef value_type& reference;typedef std::size_t size_type;typedef ptrdiff_t difference_type; public:typedef __deque_iterator<T,T&,T*,BufSiz> iterator;protected: //Internal typedefs//元素的指針的指針 (pointer of pointer of T)typedef pointer* map_pointer;protected: //Data membersmap_pointer map; //指向map,map是塊連續空間,其內部的每個元素都是一個指針(稱為節點),//指向的是一塊緩沖區std::size_t map_size; //map內可以容納多少指針iterator start; //指向的是第一個節點iterator finish; //指向的是最后一個節點 public: //Basic accessorsiterator begin(){ return start; }iterator end(){ return finish; }reference operator[](size_type n){//調用__deque_iterator<>::operator[]return start[difference_type(n)];}//調用__deque_iterator<>::operator*reference front(){return *start;}reference back(){iterator tmp = finish;--tmp; //調用__deque_iterator<>::operator--return *tmp; //調用__deque_iterator<>::operator*/** 以上三行為何不使用 return *(finish-1); 替代* 書上給出的理由是:因為__deque_iterator<>沒有為(finish-1)定義運算子*/}//調用__deque_iterator<>::operator-size_type size() const{return finish - start;}//不理解size_type max_size()const{return size_type (-1);}bool empty(){return finish == start;} };

?

//負責產生并安排好deque的結構void create_map_and_nodes(size_type num_elements){//需要節點的個數 = (元素的個數 / 每個緩沖區容納的元素的個數) + 1;//如果剛好整除 會多為其分配一=一個節點size_type num_nodes = num_elements / buffer_size() + 1;//一個map需要管理幾個節點。最少是8個,最多是 "所需要的節點數 + 2"//前后各預留一個 擴充的時候使用map_size = std::max(initial_map_size(),num_nodes + 2);//配置一個具有map_size的個節點的mapmap = map_allocator::allocate(map_size);//以下是讓nstart和nfinish指向的map所擁有的全部節點的最中央區段//保持在最中央的目的是為了保證頭尾兩端擴充的能量一樣大,每個節點對應一個緩沖區map_pointer nstart = map + (map_size - num_nodes) / 2;map_pointer nfinish = nstart + num_nodes -1;map_pointer cur;try {//為map內的每一個現用節點配置緩沖區,所有緩沖區加起來就是deque的可用空間//最后一個緩沖區可能存在一些富裕for (cur = nstart;cur<=finish;++cur) {*cur = allocate_node();}} catch (std::exception ) {//使用catch捕捉異常//"commit or rollback" 如非全部成功 就一個不要保留}//為deque內的兩個迭代器start和end設定正確的內容start.set_node(nstart);finish.set_node(nfinish);start.cur = start.first; //first、cur都是public//如果剛好整除,會多分配一個節點//令cur指向這多分配的一個節點(所對映之緩沖區)的起始處finish.cur = finish.first + num_elements % buffer_size();}//push_back()void push_back(const value_type& t){if (finish.cur != finish.last - 1){//最后尚有一個以上的備用空間//直接在備用空間上進行元素的構造Chy::allocator<T>::construct(finish.cur,t);//調整最后緩沖區的使用狀態++finish.cur;} else{//最后緩沖區已經 無(或者只剩下一個)元素的備用空間push_back_aux(t);}}//當尾端只剩下一個元素作為備用的空間,于是會調用push_back_aux()//先分配一整塊新的緩沖區,再設置新的元素的內容,然后更改迭代器finish的狀態//只有當 finish.cur == finish.last - 1 的時候才會調用void push_back_aux(const value_type& t){value_type t_copy = t;reverse_map_at_back(); //符合某種條件則必須重換一個map*(finish.node + 1) = allocate_node(); //配置一個新的節點 (緩沖區)try {Chy::allocator<T>::construct(finish.cur,t_copy);//針對標的元素設定初值finish.set_node(finish.node+1);//改變finish令其指向新的節點finish.cur = finish.first; //設定finish的狀態}catch (std::exception){Chy::allocator<T>::deallocate(*(finish.node + 1));}}void push_front(const value_type& t){if (start.cur != start.first){ //第一緩沖區尚有備用的空間Chy::allocator<T>::construct(start.cur - 1,t); //直接在備用空間上構造元素--start.cur; //調整第一緩沖區的使用狀態} else{ //第一緩沖區已經無備用空間push_back_aux(t);}}//當start.cur == start.first時才會被調用//當第一個緩沖區沒有任何備用元素的時候才會被調用void push_front_aux(const value_type& t){value_type t_copy = t;reverse_map_at_front(); //如果滿足條件則必須要重新換一個map*(start.node - 1) = allocate_node(); //配置一個新的節點(緩沖區)try{start.set_node(start.node - 1); //改變start,另其指向新的節點start.cur = start.cur - 1; //設定start的狀態Chy::allocator<T>::construct(start.cur,t_copy);}catch (std::exception){//commit or rollbackstart.set_node(start.node + 1);start.cur = start.first;Chy::allocator<T>::deallocate(*(start.node - 1));throw ;}}//整治的實際操作是通過調用 reallocate_map()函數 來實現的void reallocate_map(size_type nodes_to_add,bool add_at_front){size_type old_num_nodes = finish.node - start.node + 1;size_type new_num_nodes = old_num_nodes + nodes_to_add;map_pointer new_nstart;if (map_size > 2 * new_num_nodes){new_nstart = map + (map_size - new_num_nodes)/2 + (add_at_front ? nodes_to_add : 0);if (new_nstart < start.node){std::copy(start.node,finish.node+1,new_nstart);} else{std::copy_backward(start.node,finish.node+1,new_nstart + old_num_nodes);}} else{size_type new_map_size = map_size + std::max(map_size,nodes_to_add) + 2;//配置一塊新的內存 給map使用map_pointer new_map = map_allocator::allocate(new_map_size);new_nstart = new_map + (new_map_size - new_num_nodes)/2 + (add_at_front ? nodes_to_add : 0);//將原map內容拷貝過來std::copy(start.node,finish.node+1,new_nstart);//釋放原mapmap_allocator::deallocate(map,map_size);map = new_map;map_size = new_map_size;}//重新設定 迭代器 start 和 finishstart.set_node(new_nstart);finish.set_node(new_nstart + old_num_nodes - 1);}//reverse_map_at_front() 和 reverse_map_at_back()負責整治 mapvoid reverse_map_at_back(size_type nodes_to_add = 1){if (nodes_to_add + 1 > map_size - (finish.node - map)){//如果map節點備用空間不足//符合上述條件則必須重新替換一個map (配置更大的內存,拷貝先前的舊的區間,釋放先前的舊的空間)reallocate_map(nodes_to_add, false);}}void reverse_map_at_front(size_type nodes_to_add = 1){if (nodes_to_add > start.node - map){//如果map的前端節點備用空間不足//符合上述條件則必須重新替換一個map (配置更大的內存,拷貝先前的舊的區間,釋放先前的舊的空間)reallocate_map(nodes_to_add, true);}}

?改編版 代碼

#include <iostream>template<class T,class Alloc> class simple_alloc{ public:static T* allocate(std::size_t n){return 0==n?0:(T*)Alloc::allocate(n * sizeof(T));}static T* allocate(void){return (T*)Alloc::allocate(sizeof (T));}static void deallocate(T* p,size_t n){if (n!=0){Alloc::deallocate(p,n * sizeof(T));}}static void deallocate(T* p){Alloc::deallocate(p,sizeof(T));} };namespace Chy{template <class T>inline T* _allocate(ptrdiff_t size,T*){std::set_new_handler(0);T* tmp = (T*)(::operator new((std::size_t)(size * sizeof (T))));if (tmp == 0){std::cerr << "out of memory" << std::endl;exit(1);}return tmp;}template<class T>inline void _deallocate(T* buffer){::operator delete (buffer);}template<class T1,class T2>inline void _construct(T1 *p,const T2& value){new(p) T1 (value); //沒看懂}template <class T>inline void _destroy(T* ptr){ptr->~T();}template <class T>class allocator{public:typedef T value_type;typedef T* pointer;typedef const T* const_pointer;typedef T& reference;typedef const T& const_reference;typedef std::size_t size_type;typedef ptrdiff_t difference_type;template<class U>struct rebind{typedef allocator<U>other;};pointer allocate(size_type n,const void * hint = 0){return _allocate((difference_type)n,(pointer)0);}void deallocate(pointer p,size_type n){_deallocate(p);}void construct(pointer p,const T& value){_construct(p,value);}void destroy(pointer p){_destroy(p);}pointer address(reference x){return (pointer)&x;}const_pointer const_address(const_reference x){return (const_pointer)&x;}size_type max_size()const{return size_type(UINT_MAX/sizeof (T));}}; }//如果是copy construction 等同于assignment而且destructor 是 trivial以下就會有效 //如果是POD型別 執行的流程就會跳轉到以下函數,這個是通過function template的參數推導機制得到的 template<class ForwardIterator,class Size,class T> inline ForwardIterator __uninitizlized_fill_n_aux(ForwardIterator first,Size n,const T&x){return fill_n(first,n,x); //交給高階函數執行 }struct __true_type{}; struct __false_type{};template<class T> struct __type_traits {typedef __true_type this_dummy_member_must_be_first;typedef __false_type has_trivial_default_constructor;typedef __false_type has_trivial_copy_constructor;typedef __false_type has_trivial_assignment_constructor;typedef __false_type has_trivial_destructor;typedef __false_type is_POD_type; };//函數的邏輯是 //首先萃取出 迭代器first的value type,然后判斷這個型別是否是POD類型 template<class ForwardIterator,class Size,class T,class T1> inline ForwardIterator __uninitizlized_fill_n(ForwardIterator first,Size n,const T&x,T1*){//以下使用的是__type_traits<T1>::is_POD_type is _PODtypedef typename __type_traits<T1>::is_POD_type is_POD;return __uninitizlized_fill_n_aux(first,n,x,is_POD()); }//如果是copy construction 等同于assignment而且destructor 是 trivial以下就會有效 //如果是POD型別 執行的流程就會跳轉到以下函數,這個是通過function template的參數推導機制得到的 template <class ForwardIterator,class T> inline void __uninitialized_fill_aux(ForwardIterator first,ForwardIterator last,const T& x,__true_type){fill(first,last,x);//調用STL算法 fill() } //如果不是POD型別 執行的流程就會轉向以下函數 這個是通過function template的參數推導機制得到的 template <class ForwardIterator,class T> inline void __uninitialized_fill_aux(ForwardIterator first,ForwardIterator last,const T& x,__false_type){ForwardIterator cur = first;//為了簡化 省略了異常處理for(;cur != last;++cur){Chy::_construct(&*cur,x);} }template<class ForwardIterator,class T,class T1> inline void __uninitialized_fill(ForwardIterator first,ForwardIterator last,const T&x,T1*){typedef typename __type_traits<T1>::is_POD_type is_POD;__uninitialized_fill_aux(first,last,x,is_POD()); }/** 迭代器first指向輸出端 (欲初始化空間) 起始處* 迭代器last指向的輸出端 (欲初始化空間) 結尾處 前閉后開區間* x 表示初始數值*/ template<class ForwardIterator,class T> ForwardIterator uninitialized_fill(ForwardIterator first,ForwardIterator last,const T&x){__uninitialized_fill(first,last,x,value_type(first)); }/** 如果n不為0,傳回n,表示buffer size由用戶自定義* 如果n為0,表示buffer size使用的是默認數值,那么* 如果sz(元素的大小,sizeof(value_type))小于512,傳回 512/sz* 如果sz不小于512,傳回1*/ inline std::size_t __deque_buf_size(std::size_t n,std::size_t sz){return n!=0 ? n : (sz < 512 ? std::size_t (512/sz):std::size_t(1)); }//雙端隊列 迭代器 template<class T,class Ref,class Ptr,std::size_t BufSiz> struct __deque_iterator{ //未繼承 std::iteratortypedef __deque_iterator<T,T&,T*,BufSiz> iterator;typedef __deque_iterator<T,const T&,const T*,BufSiz> const_iterator;static std::size_t buffer_size() {return __deque_buf_size(BufSiz,sizeof (T));}//未繼承 std::iterator,所以需要自行撰寫五個必要的迭代器對應的類別typedef std::random_access_iterator_tag iterator_category;//1typedef T value_type; //2typedef Ptr pointer; //3typedef Ref reference; //4typedef std::size_t size_type;typedef ptrdiff_t difference_type; //5typedef T** map_pointer;typedef __deque_iterator self;//保持和容器的聯結T* cur; //此迭代器所指向的緩沖區的現行(current)元素T* first; //此迭代器所指之緩沖區的頭T* last; //此迭代器所指之緩沖區的尾 (含備用空間)map_pointer node;//指向管控中心public:void set_node(map_pointer new_node){node = new_node;first = *new_node;last = first + difference_type (buffer_size());}//重載各個運算子 __deque_iterator<> 成功運作的關鍵reference operator*() const {return *cur;}pointer operator->() const {return &(operator*());}difference_type operator-(const self& x) const{return difference_type (buffer_size()) * (node - x.node - 1) +(cur - first) + (x.last - x.cur);}//參考More Effective C++//Distinguish between prefix and postfix of increment and decrement operators//注意last指向的是最后一個節點的后面,所以先遞增再判斷self& operator++(){++cur; //切換至下一個元素if (cur==last){ //如果已經到達所在緩沖區的尾端set_node(node+1);//切換至下一個節點 (亦 即緩沖區的第一個元素)cur = first;}return *this;}self& operator++(int){ //后置式 標準寫法self tmp = *this;++*this; //調用self& operator++()return tmp;}//first指向的是最后一個節點,所以需要先進行判斷self& operator--(){if (cur == first){ //如果已經到達所在緩沖區的頭端set_node(node-1);//切換至前一個節點(亦即緩沖區)的最后一個元素cur = last;}--cur; //切換至前一個元素return *this;}self& operator--(int){ //后置式 標準寫法self tmp = *this;--*this;return tmp;}//以下代碼實現隨機存取,迭代器直接跳過 n 個距離self& operator+=(difference_type n){difference_type offset = n + (cur - first);if (offset >= 0 && offset < difference_type(buffer_size())){//目標位置在同一緩沖區內cur += n;} else{//目標位置不在同一個緩沖區內difference_type node_offset =offset > 0 ? offset / difference_type(buffer_size()): -difference_type ((-offset - 1) / buffer_size()) - 1;//切換到正確的節點(即對應的緩沖區)set_node(node+node_offset);//切換至正確的元素cur = first+(offset - node_offset * difference_type(buffer_size()));}return *this;}//參考More Effective C++ item22//consider using op= instead of stand-alone opself operator+(difference_type n) const{self tmp = *this;return tmp += n; //調用operator+=}//通過使用operator+= 完成operator-=self& operator-=(difference_type n){return *this += -n;}self operator-(difference_type n)const{self tmp = *this;return tmp -= n;//調用operator-=}//以下實現隨機存取,迭代器可以直接跳躍n個距離reference operator[](difference_type n)const {//以上調用operator* ,operator+return *(*this + n);}bool operator==(const self& x)const{ return cur == x.cur;}bool operator!=(const self& x)const{ return cur != x.cur;}bool operator< (const self& x)const{return (node == x.node) ? (cur < x.cur) : (node < x.node);} };//雙端隊列 template<class T,class Alloc,std::size_t BufSiz = 0> class deque{ public: //Basic typestypedef T value_type;typedef value_type* pointer;typedef value_type& reference;typedef std::size_t size_type;typedef ptrdiff_t difference_type;public:typedef __deque_iterator<T,T&,T*,BufSiz> iterator;protected: //Internal typedefs//元素的指針的指針 (pointer of pointer of T)typedef pointer* map_pointer;protected: //Data membersmap_pointer map; //指向map,map是塊連續空間,其內部的每個元素都是一個指針(稱為節點),//指向的是一塊緩沖區std::size_t map_size; //map內可以容納多少指針iterator start; //指向的是第一個節點iterator finish; //指向的是最后一個節點public: //Basic accessorsiterator begin(){ return start; }iterator end(){ return finish; }reference operator[](size_type n){//調用__deque_iterator<>::operator[]return start[difference_type(n)];}//調用__deque_iterator<>::operator*reference front(){return *start;}reference back(){iterator tmp = finish;--tmp; //調用__deque_iterator<>::operator--return *tmp; //調用__deque_iterator<>::operator*/** 以上三行為何不使用 return *(finish-1); 替代* 書上給出的理由是:因為__deque_iterator<>沒有為(finish-1)定義運算子*/}//調用__deque_iterator<>::operator-size_type size() const{return finish - start;}//不理解size_type max_size()const{return size_type (-1);}bool empty(){return finish == start;}protected: //Internal typedef//專屬配置器 每次配置一個元素的大小typedef simple_alloc<value_type,Alloc>data_allocator;//專屬配置器 每次配置一個指針大小typedef simple_alloc<pointer,Alloc>map_allocator;static size_type initial_map_size() { return 8; }static size_type buffer_size() {//返回return __deque_buf_size(BufSiz, sizeof(value_type));}void allocate_node() { return data_allocator::allocate(buffer_size()); }void deallocate_node() {return data_allocator::deallocate(buffer_size());}//構造器deque(int n,const value_type& value):start(),finish(),map(0),map_size(0){fill_initialize(n,value);} protected://負責產生并安排好deque的結構,并設定元素數值void fill_initialize(size_type n,const value_type& value){//把deque的結構轉備好create_map_and_nodes(n);map_pointer cur;try {//為每個節點的緩沖區設定初始值for (cur = start.node;cur < finish.node;++cur) {uninitialized_fill(*cur,*cur + buffer_size(),value);}//最后一個節點的設定稍微有些不同(因為尾端可能有備用空間,不必為其設定初值)uninitialized_fill(finish.first,finish,cur,value);}catch (std::exception) {}}//負責產生并安排好deque的結構void create_map_and_nodes(size_type num_elements){//需要節點的個數 = (元素的個數 / 每個緩沖區容納的元素的個數) + 1;//如果剛好整除 會多為其分配一=一個節點size_type num_nodes = num_elements / buffer_size() + 1;//一個map需要管理幾個節點。最少是8個,最多是 "所需要的節點數 + 2"//前后各預留一個 擴充的時候使用map_size = std::max(initial_map_size(),num_nodes + 2);//配置一個具有map_size的個節點的mapmap = map_allocator::allocate(map_size);//以下是讓nstart和nfinish指向的map所擁有的全部節點的最中央區段//保持在最中央的目的是為了保證頭尾兩端擴充的能量一樣大,每個節點對應一個緩沖區map_pointer nstart = map + (map_size - num_nodes) / 2;map_pointer nfinish = nstart + num_nodes -1;map_pointer cur;try {//為map內的每一個現用節點配置緩沖區,所有緩沖區加起來就是deque的可用空間//最后一個緩沖區可能存在一些富裕for (cur = nstart;cur<=finish;++cur) {*cur = allocate_node();}} catch (std::exception ) {//使用catch捕捉異常//"commit or rollback" 如非全部成功 就一個不要保留}//為deque內的兩個迭代器start和end設定正確的內容start.set_node(nstart);finish.set_node(nfinish);start.cur = start.first; //first、cur都是public//如果剛好整除,會多分配一個節點//令cur指向這多分配的一個節點(所對映之緩沖區)的起始處finish.cur = finish.first + num_elements % buffer_size();}//push_back()void push_back(const value_type& t){if (finish.cur != finish.last - 1){//最后尚有一個以上的備用空間//直接在備用空間上進行元素的構造Chy::allocator<T>::construct(finish.cur,t);//調整最后緩沖區的使用狀態++finish.cur;} else{//最后緩沖區已經 無(或者只剩下一個)元素的備用空間push_back_aux(t);}}//當尾端只剩下一個元素作為備用的空間,于是會調用push_back_aux()//先分配一整塊新的緩沖區,再設置新的元素的內容,然后更改迭代器finish的狀態//只有當 finish.cur == finish.last - 1 的時候才會調用void push_back_aux(const value_type& t){value_type t_copy = t;reverse_map_at_back(); //符合某種條件則必須重換一個map*(finish.node + 1) = allocate_node(); //配置一個新的節點 (緩沖區)try {Chy::allocator<T>::construct(finish.cur,t_copy);//針對標的元素設定初值finish.set_node(finish.node+1);//改變finish令其指向新的節點finish.cur = finish.first; //設定finish的狀態}catch (std::exception){deallocate_node(*(finish.node + 1));}}void push_front(const value_type& t){if (start.cur != start.first){ //第一緩沖區尚有備用的空間Chy::allocator<T>::construct(start.cur - 1,t); //直接在備用空間上構造元素--start.cur; //調整第一緩沖區的使用狀態} else{ //第一緩沖區已經無備用空間push_back_aux(t);}}//當start.cur == start.first時才會被調用//當第一個緩沖區沒有任何備用元素的時候才會被調用void push_front_aux(const value_type& t){value_type t_copy = t;reverse_map_at_front(); //如果滿足條件則必須要重新換一個map*(start.node - 1) = allocate_node(); //配置一個新的節點(緩沖區)try{start.set_node(start.node - 1); //改變start,另其指向新的節點start.cur = start.cur - 1; //設定start的狀態Chy::allocator<T>::construct(start.cur,t_copy);}catch (std::exception){//commit or rollbackstart.set_node(start.node + 1);start.cur = start.first;deallocate_node(*(start.node - 1));throw ;}}//整治的實際操作是通過調用 reallocate_map()函數 來實現的void reallocate_map(size_type nodes_to_add,bool add_at_front){size_type old_num_nodes = finish.node - start.node + 1;size_type new_num_nodes = old_num_nodes + nodes_to_add;map_pointer new_nstart;if (map_size > 2 * new_num_nodes){new_nstart = map + (map_size - new_num_nodes)/2 + (add_at_front ? nodes_to_add : 0);if (new_nstart < start.node){std::copy(start.node,finish.node+1,new_nstart);} else{std::copy_backward(start.node,finish.node+1,new_nstart + old_num_nodes);}} else{size_type new_map_size = map_size + std::max(map_size,nodes_to_add) + 2;//配置一塊新的內存 給map使用map_pointer new_map = map_allocator::allocate(new_map_size);new_nstart = new_map + (new_map_size - new_num_nodes)/2 + (add_at_front ? nodes_to_add : 0);//將原map內容拷貝過來std::copy(start.node,finish.node+1,new_nstart);//釋放原mapmap_allocator::deallocate(map,map_size);map = new_map;map_size = new_map_size;}//重新設定 迭代器 start 和 finishstart.set_node(new_nstart);finish.set_node(new_nstart + old_num_nodes - 1);}//reverse_map_at_front() 和 reverse_map_at_back()負責整治 mapvoid reverse_map_at_back(size_type nodes_to_add = 1){if (nodes_to_add + 1 > map_size - (finish.node - map)){//如果map節點備用空間不足//符合上述條件則必須重新替換一個map (配置更大的內存,拷貝先前的舊的區間,釋放先前的舊的空間)reallocate_map(nodes_to_add, false);}}void reverse_map_at_front(size_type nodes_to_add = 1){if (nodes_to_add > start.node - map){//如果map的前端節點備用空間不足//符合上述條件則必須重新替換一個map (配置更大的內存,拷貝先前的舊的區間,釋放先前的舊的空間)reallocate_map(nodes_to_add, true);}}void pop_back(){if (finish.cur != finish.first){//最后一個緩沖區存在一個 或者 更多的元素--finish.cur; //調整指針,相當于排除了最后的元素Chy::allocator<T>::destroy(finish.cur); //析構最后的元素} else{pop_back_aux();}}//finish.cur == finish.firstvoid pop_back_aux(){deallocate_node(finish.first); //釋放最后一個緩沖區finish.set_node(finish.node - 1);//調整finish的狀態finish.cur = finish.last - 1;//將finish的cur指針指向最后一個元素Chy::allocator<T>::destroy(finish.cur);//進行元素的析構}void pop_front(){if (start != start.last - 1){//第一緩沖區有一個或者多個元素Chy::allocator<T>::destroy(start.cur);//析構當前的元素++start.cur; //調整指針,相當于排除了第一個元素} else{pop_back_aux(); //這里將進行緩沖區的釋放工作}}//當start.cur == start.last-1被調用void pop_front_aux(){Chy::allocator<T>::destroy(start.cur); //將第一緩沖區僅剩的一個元素析構deallocate_node(start.first); //釋放第一個緩沖區start.set_node(start.node + 1);//調整start的狀態start.cur = start.first; //將start指向緩沖區的第一個元素}//清除整個deque//需要注意的是 deque的最初狀態(無任何元素的時候) 保證有一個緩沖區void clear(){//針對頭尾指針以外的每一個緩沖區,都應該是飽滿的for(map_pointer node = start.node + 1;node < finish.node - 1;++node){//將緩沖區內的所有元素全部析構//調用destroy()的第二版本Chy::allocator<T>::deallocate(*node,buffer_size());}if (start.node != finish.node){//至少有頭尾兩個緩沖區Chy::allocator<T>::destroy(start.cur,start.last); //將頭緩沖區目前所有的元素析構Chy::allocator<T>::destroy(finish.first,finish.cur); //將尾緩沖區目前所有的元素析構//釋放尾緩沖區的的物理內存,但是保留頭部緩沖區data_allocator::deallocate(finish.first,buffer_size());} else{//只有一個緩沖區//將唯一緩沖區內的所有元素析構//注意:并不釋放緩沖區的空間,將這唯一的緩沖區保留Chy::allocator<T>::destroy(start.cur,finish.cur);}//調整狀態finish = start;}//使用erase消除特定位置上的元素//消除pos指向的元素,pos為清除點iterator erase(iterator pos){iterator next = pos;++next;difference_type index = pos - start; //清除點之前的元素的個數if(index < (size() >> 1)){ //如果清除點之前的元素比較少std::copy_backward(start,pos,next);//移動清除點之前的元素pop_front(); //移動完畢,將最前一個元素刪除} else{//清除點之后的元素較少std::copy(next,finish,pos);//移動清除點之后的元素pop_back();//移動完畢 將最后一個冗余的元素刪除}return start+index;}//使用erase刪除[first,last)區間內的所有元素// 也就是刪除指定范圍內的一段元素iterator erase(iterator first,iterator last){if (first == start && last == finish){//如果刪除的是整段內存空間,直接調用clear()clear();return finish;} else{difference_type n = last - first; //清除區間的長度difference_type elements_before = first - start; //清除區間前方的元素個數if (elements_before < (size() - n)/2){ //如果前方的元素比較少//copy_backward和copy函數類似,只是逆向對元素進行復制std::copy_backward(start,first,last); //向后移動前方的元素(覆蓋清除區間)iterator new_start = start + n; //標記deque的新的起點Chy::allocator<T>::destroy(start,new_start); //移動完畢之后,將冗余的元素析構//以下將冗余的緩沖區釋放for (map_pointer cur = start.node; cur < new_start.node;++cur){data_allocator::deallocate(*cur,buffer_size());}start = new_start;//設定deque的新起點} else{//如果刪除區間的后方的元素比較少std::copy(last,finish,first); //向前移動后方的元素 (覆蓋清除區間)iterator new_finish = finish - n; //標記finish的新的尾巴Chy::allocator<T>::destroy(new_finish,finish); //移動完畢,將冗余的元素進行析構//以下將釋放冗余的緩沖區for(map_pointer cur = new_finish.node +1;cur <= finish.node;++cur){data_allocator::deallocate(cur,buffer_size());}//設定新的尾節點finish = new_finish;}return start + elements_before;}}//在某個點之前插入一個元素,并設定其值iterator insert(iterator position,const value_type& x){if (position.cur == start.cur){ //如果插入點是deque的最前端push_front(x);return start;}else if(position.cur == finish.cur){ //如果插入點是deque的最后端pop_back(x);iterator tmp = finish;--tmp;return tmp;} else{return insert_aux(position,x);}}iterator insert_aux(iterator pos,const value_type& x){difference_type index = pos - start;//插入點之前的元素的個數value_type x_copy = x;if (index < (size() / 2)){//如果插入節點之前的元素的個數比較少push_front(front());//在最前端加上和第一個元素數值相同的元素iterator front1 = start;//以下標記記號,然后進行元素的移動++front1;iterator front2 = front1;++front2;pos = start + index;iterator pos1 = pos;++pos1;std::copy(front2,pos1,front1);//元素的移動} else{push_back(back());iterator back1 = finish;--back1;iterator back2 = back1;--back2;pos = start + index;std::copy_backward(pos,back2,back1);}//在插入點上設定新值*pos = x_copy;return pos;}};

?參考鏈接

  • STL之deque實現詳解_一個菜鳥的博客-CSDN博客_deque
  • C++ copy_backward(STL copy_backward)算法詳解

總結

以上是生活随笔為你收集整理的STL源码剖析 deque双端队列 概述的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

又粗又长又大又爽又黄少妇毛片 | 免费黄色在线网址 | 日韩试看| 在线看片一区 | 亚洲人久久 | 在线中文字母电影观看 | 人人爽人人av | 精品资源在线 | 成人免费看视频 | 色网站免费在线观看 | 国产无吗一区二区三区在线欢 | 网站在线观看你们懂的 | 久色伊人| 热久精品 | 国产一区二区在线视频观看 | 九九久久国产精品 | 国产黄免费看 | 国产精品理论片在线播放 | 日韩精品在线免费观看 | 91久久偷偷做嫩草影院 | 一级一片免费看 | 国产男女无遮挡猛进猛出在线观看 | 99久久久久久久久 | www.夜夜操.com| 成人av久久 | 日韩免费在线看 | 98超碰人人 | 久久亚洲影视 | 91网站观看| 操操操日日日干干干 | 精品v亚洲v欧美v高清v | 国产精品激情偷乱一区二区∴ | 狠狠躁夜夜躁人人爽超碰97香蕉 | 九九在线精品视频 | 午夜电影久久久 | 国产精品嫩草影院99网站 | 在线观看免费黄色 | 免费情缘| 精品国产一区二区三区久久久久久 | 精品久久一区二区 | 日韩在线不卡视频 | 天天草网站 | 国产精品乱码高清在线看 | 黄色av高清 | 国产精品一区二区久久久 | 国产亚洲精品久久久久久网站 | 亚洲精品国产精品久久99 | 成人毛片一区 | 99九九热只有国产精品 | 在线观看 亚洲 | 欧美成年人在线视频 | 久久久久久久久久久精 | 91豆花在线观看 | 国产69久久精品成人看 | 99精品亚洲 | 欧美精品在线视频 | 欧美一级片 | 国产精品亚洲人在线观看 | 少妇bbbb搡bbbb搡bbbb | 99久久久久久久 | 男女精品久久 | 五月婷婷国产 | 最新午夜电影 | 久久亚洲福利视频 | 97在线观看免费视频 | 精品一区二区免费视频 | 久久久久国产成人免费精品免费 | 福利视频网址 | 精品国产不卡 | 天天干天天操天天爱 | 午夜精品久久久久久久99 | 狠狠干我| 国产97碰免费视频 | 国产破处视频在线播放 | 色婷婷骚婷婷 | 亚洲人久久 | 人人插人人做 | 深爱五月激情五月 | 六月色婷婷 | 免费高清在线视频一区· | 久久精品—区二区三区 | 在线看片视频 | 综合天堂av久久久久久久 | 9在线观看免费高清完整版 玖玖爱免费视频 | 欧美a级片免费看 | 999久久久久久久久久久 | 国产成人精品一区二三区 | 在线a视频免费观看 | 一区二区三区在线观看 | 97精品免费视频 | 天天操夜夜想 | 97超碰超碰久久福利超碰 | 久久韩国免费视频 | 国产视频九色蝌蚪 | 黄色大全视频 | 视频一区亚洲 | 久久精品1区 | 91综合色 | 97免费在线观看视频 | 香蕉色综合| 成人久久 | 婷婷草| 五月天婷婷视频 | 久久小视频 | 国产精品久久在线观看 | 日本mv大片欧洲mv大片 | 青青河边草手机免费 | 久久夜夜夜| 在线观看不卡视频 | 国产精品国产亚洲精品看不卡15 | 日日久视频 | 日本一区二区三区免费观看 | 91成人精品一区在线播放 | 人人插人人草 | 国产一级a毛片视频爆浆 | 一区二区久久 | 欧美乱码精品一区二区 | 国产精品一区二区久久精品爱微奶 | 欧美日韩在线电影 | 91福利影院在线观看 | 欧美a视频在线观看 | 日日久视频 | 91入口在线观看 | 超碰97.com| 中文字幕久久精品 | 久久国产片 | 综合婷婷丁香 | 精品国产区 | 免费在线观看日韩视频 | 国产精品va | 久久激情视频免费观看 | 亚洲永久精品在线 | 成人免费观看大片 | 精品高清视频 | 豆豆色资源网xfplay | 国产在线观看高清视频 | 日本女人逼 | 欧美日韩免费一区 | 在线免费视 | 亚洲视频2 | 一区二区三区四区免费视频 | 久久久网页 | 九九激情视频 | 综合网久久 | 国产一级淫片在线观看 | 国产精品久久久久久99 | 国产精品一区在线观看 | 涩涩色亚洲一区 | 亚洲综合在线视频 | 欧美一级性 | av专区在线 | 精品女同一区二区三区在线观看 | 久久99久久99精品免观看软件 | 日韩大片在线免费观看 | 狠狠色综合欧美激情 | 91大神免费视频 | 夜夜爽夜夜操 | 欧美 亚洲 另类 激情 另类 | 91手机视频在线 | 国产精久久久久久久 | 91黄色在线视频 | 日韩精品视频在线免费观看 | 一区二区三区免费网站 | 国产区av在线 | 精品国产免费久久 | 日韩在线免费视频 | 亚洲成人资源在线观看 | 欧美a级成人淫片免费看 | 日韩精品中文字幕久久臀 | 婷婷av色综合 | www.色午夜 | 美女免费av| 91福利社区在线观看 | 97精品国产97久久久久久粉红 | 91成人精品一区在线播放69 | 日韩高清一 | 五月天婷婷丁香花 | 波多野结衣资源 | 中文字幕日韩国产 | 中文字幕久久久精品 | 亚洲精品视频网站在线观看 | 狠狠色丁香婷婷综合视频 | 色婷婷国产精品一区在线观看 | 在线高清av| 在线观看免费91 | 亚洲第一色 | 夜夜天天干 | 99精品在线免费在线观看 | 亚洲精品视频免费在线 | 黄色a视频免费 | 亚洲在线精品 | 欧美 亚洲 另类 激情 另类 | 欧美一区二区在线免费观看 | 精品一区精品二区 | 99re8这里有精品热视频免费 | 亚洲男男gaygay无套 | 亚洲欧美激情插 | 麻豆94tv免费版 | 91精品久久久久久综合五月天 | 免费看污网站 | 欧美巨乳网 | 97人人添人澡人人爽超碰动图 | 九九色综合 | 在线播放亚洲激情 | 一区电影 | 99精品视频免费看 | 中文字幕在线视频免费播放 | 国产精品久久99 | 夜夜躁狠狠燥 | 91爱爱免费观看 | 婷婷六月丁 | 99欧美视频| 制服丝袜一区二区 | 国产+日韩欧美 | 亚洲欧美国产精品va在线观看 | 97精品国产 | av免费看av | 高清av免费看 | 亚洲第一区在线播放 | 精品久久久久久国产偷窥 | 99九九热只有国产精品 | 成人久久精品视频 | 亚洲成人av免费 | 国产二区av| 中文字幕亚洲不卡 | 久久久人人人 | 国产在线不卡精品 | 色偷偷88888欧美精品久久 | 国产成人三级一区二区在线观看一 | 91精品日韩| 蜜桃av久久久亚洲精品 | 五月色综合 | 国产69精品久久app免费版 | 国产专区视频 | 黄a网| 狠狠色丁香婷婷 | 久久久久伊人 | 在线成人国产 | 天天操网址 | 欧美激情精品久久久久久免费印度 | 亚洲色综合 | 国产精品嫩草影视久久久 | 天天综合色 | 在线精品观看国产 | 久久久黄视频 | 久久久噜噜噜久久久 | 又黄又刺激视频 | 亚洲理论影院 | 天天曰夜夜操 | 91人人揉日日捏人人看 | 91桃色免费观看 | 91九色蝌蚪视频网站 | 啪啪免费观看网站 | 色婷婷狠狠五月综合天色拍 | 国产美女精品视频 | 免费91麻豆精品国产自产在线观看 | 久久精品99久久久久久2456 | 国产午夜av | 欧美日韩三级 | 日韩色中色 | 久久国产精品视频免费看 | 成人av电影网址 | 香蕉97视频观看在线观看 | 国产高清免费av | 日本h在线播放 | 91最新在线| 高清视频一区 | 黄网站色成年免费观看 | 国产一级免费片 | 亚洲国产片 | 国产一区网 | japanesefreesexvideo高潮 | av大全在线看 | 亚洲一区二区黄色 | 日韩av一区在线观看 | 亚洲一区二区精品在线 | 在线观看中文字幕2021 | 亚洲精品一区中文字幕乱码 | 91完整版在线观看 | 色的网站在线观看 | 不卡的av在线播放 | 色婷婷一区 | 日韩欧美精品免费 | 国内精品久久久久久中文字幕 | 久久99精品视频 | 亚洲高清免费在线 | 日韩av二区| 欧美精品午夜 | 天天摸日日操 | 亚洲专区中文字幕 | av天天澡天天爽天天av | 精品人人人 | 国语精品视频 | 深爱开心激情 | 日韩成人免费电影 | 亚洲精品久久久蜜桃 | 在线观看av大片 | 91精品国产九九九久久久亚洲 | 91精品国产91久久久久福利 | 国产高清日韩 | 91av资源网 | 成人小视频在线观看免费 | 欧美激情综合五月色丁香 | 久久久久网址 | 欧美成人影音 | 婷婷综合在线 | 99精品在线观看 | 欧美日韩性视频在线 | 日韩精品免费在线播放 | 美女一区网站 | 国产永久网站 | 久久久这里有精品 | 久久综合射| 江苏妇搡bbbb搡bbbb | 久久免费视频观看 | 国产直播av | 黄色小说免费在线观看 | 亚洲国产成人高清精品 | 国产欧美综合在线观看 | 丁香久久五月 | 欧美日本中文字幕 | 久草在线久 | 欧美小视频在线观看 | 丁香六月天 | 安徽妇搡bbbb搡bbbb | 久久手机视频 | 国产精品高清免费在线观看 | www.黄色网.com | 狠狠色丁香婷婷综合橹88 | 在线免费观看国产精品 | 黄色a在线 | 天天操天天舔天天爽 | 一级成人免费视频 | av电影在线观看完整版一区二区 | 亚洲精品美女久久 | 97高清视频| 国产精品久久久久婷婷二区次 | 麻豆精品在线视频 | 国产精品自拍在线 | 超碰97国产在线 | 国产日韩在线看 | 日韩午夜电影网 | 久久久久久国产精品久久 | 欧美激情精品久久久久久变态 | 久久国产精品99久久久久久丝袜 | 亚洲成人在线免费 | 正在播放五月婷婷狠狠干 | 色婷婷播放 | 婷婷丁香花五月天 | 欧美日韩裸体免费视频 | 国产视频一区二区在线 | 中文字幕乱在线伦视频中文字幕乱码在线 | 亚洲在线视频观看 | 久久三级视频 | 999ZYZ玖玖资源站永久 | www久久精品| 一区二区三区污 | 中文字幕一区二区在线观看 | 青草视频免费观看 | 亚洲日本va午夜在线影院 | 日韩av中文字幕在线 | 欧美视频网址 | 最近中文字幕完整高清 | 久久在线免费观看视频 | 国产成人免费在线观看 | 亚洲综合视频在线 | 在线中文字幕一区二区 | 日日躁夜夜躁xxxxaaaa | 狠狠干.com| 97超碰人人 | 亚洲日日射 | 婷婷丁香自拍 | www日韩高清 | 日韩av免费一区 | 成人黄大片 | 亚洲理论在线观看电影 | 五月激情婷婷丁香 | 精品在线观看一区二区三区 | 91视频-88av | 麻豆影音先锋 | 96av麻豆蜜桃一区二区 | 成人黄性视频 | 国内精品国产三级国产aⅴ久 | 色婷婷免费视频 | 五月开心六月伊人色婷婷 | 福利片视频区 | www好男人 | 亚洲成年人在线播放 | 亚洲理论影院 | 九九九九色 | 久草国产视频 | 精品国产一区二区三区av性色 | 国产视频在线观看一区 | 国产美女精品在线 | 久久久久欠精品国产毛片国产毛生 | 99在线视频免费观看 | 在线观看中文字幕亚洲 | 日本精油按摩3 | 一级特黄av | 久久精品波多野结衣 | 欧美三级在线播放 | 久久狠狠干| 狠狠干电影 | 久久福利 | 欧美极品xxx | 午夜性生活 | 综合网久久 | 狠狠婷婷 | 91中文字幕在线播放 | 狠狠躁夜夜a产精品视频 | 日韩欧美视频在线观看免费 | 日日操日日插 | 国产精品久久久久久久免费大片 | 丁香激情视频 | 亚洲一区日韩 | 久久久亚洲影院 | 成人av地址 | 91视频链接| 六月婷婷网| 中文不卡视频 | 久久伊人精品一区二区三区 | 色视频在线免费 | 国产一级视频在线观看 | 91精品影视 | 成人黄色大片在线免费观看 | 免费观看9x视频网站在线观看 | 亚洲高清免费在线 | 国产精品美女久久久久久久 | 天天操一操| 91天堂素人约啪 | 久久婷婷国产 | 中文字幕在线高清 | 久艹在线观看视频 | 麻豆免费观看视频 | 在线观看视频色 | 久久调教视频 | 欧美日高清视频 | 伊人亚洲综合 | 久久九九网站 | 亚洲一区不卡视频 | 探花视频在线版播放免费观看 | 久久99热这里只有精品国产 | 精品视频网站 | 久久久久久久久久国产精品 | 中文字幕精品一区二区精品 | 九九九九九精品 | 欧美日韩中 | 天天综合色天天综合 | 亚洲三级在线播放 | 88av视频| 欧美黄网站 | 久久成人精品电影 | 国产理论一区二区三区 | 青草草在线 | 超碰.com| 亚洲成人网在线 | 999色视频| 深爱婷婷网 | 欧美狠狠色 | 综合黄色网 | 超碰夜夜 | 91热精品 | 狠狠激情中文字幕 | 综合五月| 亚洲精品乱码久久久久久蜜桃动漫 | 日韩免费专区 | 国产高清专区 | 精选久久 | 男女激情免费网站 | 成人av片免费看 | 超碰成人免费电影 | 日韩深夜在线观看 | 激情xxxx | 欧洲一区二区在线观看 | 亚洲精品美女在线观看播放 | 一级黄色a视频 | 亚洲国产成人精品在线观看 | 激情久久一区二区三区 | 日韩乱色精品一区二区 | 亚洲涩综合 | 亚洲午夜精品久久久久久久久久久久 | av资源免费在线观看 | 在线最新av | 亚州精品在线视频 | 亚洲精品色婷婷 | 国产一在线精品一区在线观看 | 黄污视频大全 | 在线国产欧美 | a天堂最新版中文在线地址 久久99久久精品国产 | 亚洲综合成人在线 | 欧美aa一级片 | 日韩欧美69 | 五月婷婷丁香 | 蜜桃av观看| 综合久久一本 | 久久蜜臀av | 天天av天天 | 免费无遮挡动漫网站 | 69人人| 国模吧一区 | 亚洲激情在线观看 | 国产精品9区 | 国产精品岛国久久久久久久久红粉 | 99久久久久国产精品免费 | 天天射天天爱天天干 | 国产视频在线观看一区 | 色姑娘综合网 | 欧美成人精品三级在线观看播放 | 国产流白浆高潮在线观看 | 亚洲午夜av电影 | 久久夜色精品国产欧美一区麻豆 | 亚洲电影免费 | 操处女逼| 99在线热播精品免费99热 | 成人av电影免费在线播放 | 国产精品岛国久久久久久久久红粉 | 精品国产一区二区三区av性色 | 亚洲人av免费网站 | 亚洲免费公开视频 | 黄色app网站在线观看 | www.色在线| 91精品91 | 伊人色综合久久天天网 | 男女全黄一级一级高潮免费看 | 黄色成人毛片 | 99这里都是精品 | 国产精品美乳一区二区免费 | av在线免费播放网站 | 99在线观看免费视频精品观看 | 97精品欧美91久久久久久 | 黄色小网站在线 | 天天操天天爱天天干 | 九九热re | 深夜国产福利 | 久久久久久久亚洲精品 | 99性视频 | 亚洲影音先锋 | 91九色九色| 中文字幕在线影视资源 | 免费观看一级特黄欧美大片 | 亚洲一区二区麻豆 | 黄色网www | av电影中文字幕 | 国产精品黄色影片导航在线观看 | 国产高清在线精品 | caobi视频| 久久久久久久久久久影院 | 狠狠色噜噜狠狠狠合久 | 免费高清在线一区 | 在线观看国产福利片 | 人人网av| 午夜视频在线观看一区二区 | av高清在线 | 成年人在线免费看 | 国产日本在线 | 揉bbb玩bbb少妇bbb | 欧美成人基地 | 久久精品欧美一区 | 成年人免费在线 | 91精品视频网站 | 手机av永久免费 | 91丨九色丨国产在线 | 一本一道波多野毛片中文在线 | 成人一区二区三区在线 | 亚洲美女免费精品视频在线观看 | 麻豆av电影 | 国产 视频 高清 免费 | 日韩电影在线观看一区二区三区 | 亚洲 中文 欧美 日韩vr 在线 | www.在线观看视频 | 精品999在线观看 | a视频在线播放 | 四虎永久免费在线观看 | 日韩免费网站 | 久久久久久久久久久免费视频 | 午夜精品电影一区二区在线 | 视频在线观看99 | 日韩欧美一区二区三区在线 | 免费黄色看片 | 欧美贵妇性狂欢 | 久久9999久久免费精品国产 | 成人午夜电影网站 | 亚洲一级久久 | 97av视频| 欧美一级黄色网 | 欧美另类xxxxx | 久久成人精品视频 | 99色免费| 五月婷婷另类国产 | 九九免费在线观看视频 | 日韩视频在线不卡 | 国产美女视频一区 | 色九色 | 久久精品播放 | 国产精品久久久久久久久久久免费看 | 免费看黄20分钟 | 国产成人黄色片 | 国产专区在线看 | 一级黄色在线免费观看 | 日韩三级中文字幕 | .国产精品成人自产拍在线观看6 | av电影一区二区三区 | 91av视频在线播放 | 日韩免费视频在线观看 | 91精品第一页 | 亚洲aⅴ一区二区三区 | 最新日韩在线观看 | 亚洲经典视频在线观看 | 中文字幕在线久一本久 | 国产亚洲精品日韩在线tv黄 | 91最新网址在线观看 | 天天射天天干天天插 | 欧美日韩精品区 | 在线观看日韩av | 免费精品人在线二线三线 | 日韩区欠美精品av视频 | 一区二区三区免费网站 | 91精品视频免费观看 | 天天色天天操天天爽 | 国内精品久久久久影院优 | 色先锋资源网 | 久久99久久99精品免观看软件 | 日日夜夜精品视频天天综合网 | 国产玖玖在线 | 四虎在线免费观看 | 在线观看国产高清视频 | 97爱爱爱 | 天天插综合 | 久久九九国产视频 | 久av在线| 久久精品影片 | 久久综合久久综合九色 | 国产一区二区在线免费 | 国产精品一区在线 | 激情图片qvod | 欧美性生活久久 | 深爱激情五月网 | 1024手机在线看 | 探花视频在线观看 | 欧美精品三级在线观看 | 91精品国产99久久久久久红楼 | 亚洲精品电影在线 | 亚洲精品久久在线 | 九九热中文字幕 | 四虎影院在线观看av | 探花视频免费观看 | 精品国产自在精品国产精野外直播 | 欧美日韩视频在线一区 | 91免费试看 | 在线视频一区观看 | 在线观看色网站 | 国产一区私人高清影院 | 97人人看 | 91黄色免费网站 | 伊人五月| 97福利社 | 久草电影免费在线观看 | 久久国产精品久久w女人spa | 在线观看亚洲国产 | 在线影视 一区 二区 三区 | 国产高清视频免费最新在线 | 九九综合九九 | 黄色大全视频 | 色婷婷福利 | 99超碰在线播放 | 国产美女免费观看 | 综合久久久久久久 | 国产一级做a | 九九热在线视频免费观看 | 欧美日韩二三区 | 在线成人免费 | 精品美女国产在线 | 99精品久久久久久久 | 狠狠婷婷 | 99热在线国产精品 | 最新av在线播放 | 91在线小视频 | 精品视频在线免费观看 | 免费三级黄 | 激情网五月婷婷 | 亚洲 欧美 综合 在线 精品 | 日日夜夜综合 | 黄色软件视频网站 | 久久免费视频6 | 国产视频在线观看一区 | 五月天电影免费在线观看一区 | 国产麻豆剧果冻传媒视频播放量 | 中文字幕亚洲在线观看 | 国产精品成人在线观看 | 国产在线观看 | 在线精品在线 | a在线免费观看视频 | 91伊人久久大香线蕉蜜芽人口 | wwwwwww黄 | 亚洲精品99久久久久中文字幕 | 免费福利在线播放 | 国产亚洲午夜高清国产拍精品 | 日韩免费精品 | 亚洲电影av在线 | 中中文字幕av在线 | h视频在线看 | 欧美日视频 | 2023国产精品自产拍在线观看 | 99人成在线观看视频 | www免费 | 成人黄色片免费看 | 六月丁香婷| 在线免费观看黄色av | 国产一级免费视频 | 人人擦| 久草视频中文在线 | 国产精品自产拍在线观看网站 | 激情av资源| 中国一级片视频 | 欧美a免费| 九九色综合 | 伊人久久一区 | 99精品免费在线观看 | 狠狠干夜夜 | 亚洲精品国产第一综合99久久 | 天天操天天射天天 | 欧美一级淫片videoshd | 黄色在线观看免费网站 | 色www精品视频在线观看 | 日本激情动作片免费看 | 色偷偷人人澡久久超碰69 | 亚洲黄色影院 | 黄色三几片| 国内精品小视频 | 欧美激情视频免费看 | 亚洲在线精品 | 国产免费视频一区二区裸体 | 欧洲亚洲精品 | 成人影视片 | 色婷婷导航 | 国产成人精品福利 | 国产在线2020| 日韩av黄 | 综合国产在线观看 | 丁香婷婷激情网 | 久久成人在线 | 国模视频一区二区 | 久久手机精品视频 | 最近最新最好看中文视频 | 国产精品18久久久久久不卡孕妇 | 成人国产精品一区二区 | 国产精品久久久久久久毛片 | av成人免费在线观看 | 99欧美精品 | 日韩精品在线视频免费观看 | 成人国产精品 | 亚洲成av人片在线观看香蕉 | 免费人成在线观看 | 麻豆传媒精品 | 一区二区三区免费播放 | 久久国产精品一二三区 | 欧美日韩高清不卡 | 日韩欧美在线免费 | 国产高清在线看 | 天天干天天操人体 | 久久精品超碰 | 久草热视频 | 99se视频在线观看 | 国产精品久久久久久久久岛 | 啪啪肉肉污av国网站 | 91麻豆.com| a黄色片 | 国产激情小视频在线观看 | 狠狠躁夜夜躁人人爽超碰97香蕉 | 蜜桃传媒一区二区 | 日本韩国欧美在线观看 | 色五月色开心色婷婷色丁香 | 久久超碰网 | a天堂一码二码专区 | 日韩精品在线观看视频 | 久久亚洲精品国产亚洲老地址 | 久久一区国产 | 日韩中文字幕亚洲一区二区va在线 | 久久成人国产精品一区二区 | 日本一区二区不卡高清 | 成人黄色大片在线观看 | 中文字幕色播 | 黄色在线免费观看网站 | 美女国产免费 | 色偷偷888欧美精品久久久 | 欧美一级高清片 | 亚洲国产精品第一区二区 | 美女天天操 | 国产在线美女 | 一区三区在线欧 | 日韩精品久久久久久中文字幕8 | 亚洲精选久久 | 99999精品 | av电影中文字幕 | 亚洲激情五月 | 中文字幕在线播放一区 | 狠狠操91 | 日本中文在线播放 | 午夜视频一区二区 | 亚洲精品婷婷 | 免费观看www视频 | 国产成人高清av | 免费合欢视频成人app | 丁香九月婷婷综合 | 黄色福利网站 | 亚洲欧美日韩中文在线 | 99国内精品 | 国产男女爽爽爽免费视频 | 国产亚洲精品久久19p | 久久艹免费 | 欧美少妇的秘密 | 亚洲视频专区在线 | 免费国产亚洲视频 | 欧洲一区二区三区精品 | 久久久影院一区二区三区 | 国产一区二区免费 | 97色综合| 国产一区播放 | 三级a视频 | 久久综合中文字幕 | 亚洲精品免费看 | 狠狠搞,com | 欧美在线视频一区二区三区 | 91亚洲狠狠婷婷综合久久久 | 亚洲精品在线电影 | 免费日韩一区二区三区 | 欧美精品一区二区三区一线天视频 | 99在线免费观看 | 久久国产片 | 96av在线视频 | 午夜成人免费电影 | 久久99网站 | 99久久日韩精品免费热麻豆美女 | 免费在线观看av网址 | 99c视频高清免费观看 | 国产在线高清 | 日日干,天天干 | 丝袜少妇在线 | 久久草在线精品 | 中文字幕一区二区三区四区在线视频 | 18久久久久 | 久久精品com | 69久久久 | 91热精品视频 | 免费男女羞羞的视频网站中文字幕 | 午夜精品久久久久久 | 99热精品国产 | 成年人av在线播放 | 欧美污污网站 | 狠狠色丁香久久婷婷综 | 久久国产亚洲精品 | 国产成人久 | 日韩理论电影在线 | 婷婷网在线| 久久久久国产一区二区 | 亚洲精品mv在线观看 | 久久精品视频18 | 亚洲三级影院 | 五月天六月婷婷 | 一区二区三区中文字幕在线观看 | 精品国产乱码久久久久久1区二区 | 亚洲专区 国产精品 | 久久精品精品电影网 | 天堂素人在线 | 亚洲综合狠狠干 | 97超碰人人澡人人 | 久久九九视频 | 992tv在线观看 | 久久99国产精品视频 | 免费精品视频在线 | 玖玖在线播放 | 久久国产区 | 探花视频在线观看免费 | 国内外成人免费在线视频 | 免费av在线播放 | 一区二区三区三区在线 | 亚洲精品在线观看网站 | 国产一级在线看 | 亚洲精品在线资源 | 亚洲2019精品 | 六月色婷婷 | 91av在线电影| 欧美日韩一区二区在线观看 | 日韩高清成人在线 | 亚洲a网 | 国产电影一区二区三区四区 | 中文字幕第 | 天堂网一区二区三区 | 成人免费观看a | 一区二区三区在线不卡 | 国产成人精品国内自产拍免费看 | 在线精品视频在线观看高清 | 亚洲视频1区2区 | 草久在线观看视频 | 国产成人精品一区二区在线观看 | 国产中文伊人 | 欧美午夜寂寞影院 | 国产精品女 | 中文字幕在线中文 | 狠狠躁夜夜躁人人爽超碰97香蕉 | 天天操天天操天天 | 人人超碰人人 | 中文字幕丝袜美腿 | 国产精品69av | 日本中文在线 | 麻豆视频免费入口 | 天天拍天天操 | 在线日韩 | 久久久亚洲精品 | 亚洲精品中文在线观看 | 免费黄a| 久久伊人八月婷婷综合激情 | 欧美黄色特级片 | 97成人在线免费视频 | 黄色毛片一级片 | 日本三级国产 | 国产精品理论片在线播放 | 欧美日韩中文视频 | 狠狠色丁香婷婷综合久小说久 | 久久久影视 | 狠狠狠色丁香婷婷综合久久五月 | 久草网站 | 亚洲爱爱视频 | 久久69精品| 狠狠干天天射 | 美女视频黄免费的 | 婷婷色在线播放 | 日韩精品不卡在线观看 | 久久国产精品久久精品国产演员表 | 91看片麻豆| 精品欧美一区二区精品久久 | 操操操影院 | 久久精品中文字幕一区二区三区 | 手机成人免费视频 | www五月 | 国产一区麻豆 | 91一区啪爱嗯打偷拍欧美 | 国产在线综合视频 | 午夜色性片 | 国产精品久久久久久久99 | 天天操夜夜操天天射 | 人人人爽| 久久久久激情视频 | 欧美黑人巨大xxxxx | 国产精品国产三级在线专区 | 日韩精品视频第一页 | 国产小视频在线免费观看视频 | 久久9视频| 欧美在线视频日韩 | 日韩精品在线一区 | 女人18毛片a级毛片一区二区 | 中文字幕在线观看视频免费 | 在线免费观看视频 | 国产精品理论片在线播放 | 97人人澡人人爽人人模亚洲 | 九月婷婷人人澡人人添人人爽 | 香蕉影院在线观看 | 在线精品视频免费播放 | 国产特级毛片 | 婷婷在线观看视频 | 亚洲精品视频在线看 | 日韩资源在线观看 | 最近中文国产在线视频 | 五月网婷婷 | 欧美作爱视频 | 亚洲综合小说 | 亚洲精品免费在线视频 | 99久久国产免费,99久久国产免费大片 | 97超级碰碰碰视频在线观看 | 波多野结衣动态图 | 91视频电影 | 网站在线观看日韩 | 日韩欧美在线观看一区二区 | 日本不卡视频 | 免费在线激情视频 | 91麻豆精品国产午夜天堂 | 成人小视频在线播放 | 日韩视频一区二区在线观看 | 日韩福利在线观看 | 国内精品小视频 | 麻豆免费视频 | 国产精品av在线免费观看 | 天天射夜夜爽 | 黄色片网站免费 | 一区二区三区日韩视频在线观看 | 国产高清在线观看 | 婷婷综合成人 | 亚洲四虎在线 | 在线观看免费视频 | 97在线视频网站 | 人人爽人人爽人人爽人人爽 | 久草视频在线资源站 | 日本精品视频网站 | 免费精品国产va自在自线 | 成人在线播放视频 | 欧美日韩中文字幕视频 | 成人h电影在线观看 | 欧美国产日韩一区二区三区 |