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

歡迎訪問 生活随笔!

生活随笔

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

编程问答

bullet HashMap 内存紧密的哈希表

發布時間:2023/12/10 编程问答 50 豆豆
生活随笔 收集整理的這篇文章主要介紹了 bullet HashMap 内存紧密的哈希表 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

last modified time:2014-11-9 14:07:00

bullet 是一款開源物理引擎,它提供了碰撞檢測、重力模擬等功能,非常多3D游戲、3D設計軟件(如3D Mark)使用它作為物理引擎。

作為物理引擎,對性能的要求是非常苛刻的;bullet項目之所以可以發展到今天,非常大程度取決于它在性能上優異的表現。

翻閱bullet的源代碼就能看到非常多源代碼級別的優化。本文將介紹的HashMap就是一個典例。

bullet項目首頁:http://bulletphysics.org/

注:bullet非常多函數定義了Debug版和Release版兩個版本號。本文僅以Release版為例。


btAlignedAllocator的接口定義

btAlignedAllocator是bullet定義的一個內存分配器接口,bullet的其它數據結構都使用它來管理內存。btAlignedAllocator的定義和STL的allocator(下面稱std::allocator)類似: ///The btAlignedAllocator is a portable class for aligned memory allocations. ///Default implementations for unaligned and aligned allocations can be overridden by a custom allocator // using btAlignedAllocSetCustom and btAlignedAllocSetCustomAligned. template < typename T , unsigned Alignment > class btAlignedAllocator {typedef btAlignedAllocator< T , Alignment > self_type;public://just going down a list:btAlignedAllocator() {}/*btAlignedAllocator( const self_type & ) {}*/template < typename Other >btAlignedAllocator( const btAlignedAllocator< Other , Alignment > & ) {}typedef const T* const_pointer;typedef const T& const_reference;typedef T* pointer;typedef T& reference;typedef T value_type;pointer address ( reference ref ) const { return &ref; }const_pointer address ( const_reference ref ) const { return &ref; }pointer allocate ( size_type n , const_pointer * hint = 0 ) {(void)hint;return reinterpret_cast< pointer >(btAlignedAlloc( sizeof(value_type) * n , Alignment ));}void construct ( pointer ptr , const value_type & value ) { new (ptr) value_type( value ); }void deallocate( pointer ptr ) {btAlignedFree( reinterpret_cast< void * >( ptr ) );}void destroy ( pointer ptr ) { ptr->~value_type(); }template < typename O > struct rebind {typedef btAlignedAllocator< O , Alignment > other;};template < typename O >self_type & operator=( const btAlignedAllocator< O , Alignment > & ) { return *this; }friend bool operator==( const self_type & , const self_type & ) { return true; } }; 與std::allocator類似。btAlignedAllocator的allocate和deallocate分別負責申請和釋放內存空間,以release版編譯的btAlignedAlloc/btAlignedFree分別為: void* btAlignedAllocInternal (size_t size, int alignment);void btAlignedFreeInternal (void* ptr);#define btAlignedAlloc(size,alignment) btAlignedAllocInternal(size,alignment)#define btAlignedFree(ptr) btAlignedFreeInternal(ptr)而btAlignedAllocInternal/btAlignedFreeInternal及其定制化的實現為: static btAlignedAllocFunc *sAlignedAllocFunc = btAlignedAllocDefault; static btAlignedFreeFunc *sAlignedFreeFunc = btAlignedFreeDefault;void btAlignedAllocSetCustomAligned(btAlignedAllocFunc *allocFunc, btAlignedFreeFunc *freeFunc) {sAlignedAllocFunc = allocFunc ? allocFunc : btAlignedAllocDefault;sAlignedFreeFunc = freeFunc ?

freeFunc : btAlignedFreeDefault; } void* btAlignedAllocInternal (size_t size, int alignment) { gNumAlignedAllocs++; // 和gNumAlignedFree結合用來檢查內存泄露 void* ptr; ptr = sAlignedAllocFunc(size, alignment); // printf("btAlignedAllocInternal %d, %x\n",size,ptr); return ptr; } void btAlignedFreeInternal (void* ptr) { if (!ptr) { return; } gNumAlignedFree++; // 和gNumAlignedAllocs 結合用來檢查內存泄露 // printf("btAlignedFreeInternal %x\n",ptr); sAlignedFreeFunc(ptr); }


如上,bullet內存分配的定制操作并不復雜。僅僅需調用下面兩個函數就可以: // The developer can let all Bullet memory allocations go through a custom memory allocator, using btAlignedAllocSetCustom void btAlignedAllocSetCustom(btAllocFunc *allocFunc, btFreeFunc *freeFunc);// If the developer has already an custom aligned allocator, then btAlignedAllocSetCustomAligned can be used. // The default aligned allocator pre-allocates extra memory using the non-aligned allocator, and instruments it. void btAlignedAllocSetCustomAligned(btAlignedAllocFunc *allocFunc, btAlignedFreeFunc *freeFunc);

不管是否定制自己的Alloc/Free(或AllignedAlloc/AlignedFree),bullet內的其它數據結構都使用btAlignedAllocator作為內存分配(回收)的接口。隨后將會看到。btAlignedAllocator的定制化設計與std::allocator的不同。文末具體討論。


btAlignedAllocator的內存對齊

btAlignedAllocator除了定制化與std::allocator不同外,還添加了內存對齊功能(從它的名字也能看得出來)。

繼續查看btAlignedAllocDefault/btAlignedFreeDefault的定義(btAlignedAllocator.{h|cpp})能夠看到:

#if defined (BT_HAS_ALIGNED_ALLOCATOR) #include <malloc.h> static void *btAlignedAllocDefault(size_t size, int alignment) {return _aligned_malloc(size, (size_t)alignment); // gcc 提供了 }static void btAlignedFreeDefault(void *ptr) {_aligned_free(ptr); } #elif defined(__CELLOS_LV2__) #include <stdlib.h>static inline void *btAlignedAllocDefault(size_t size, int alignment) {return memalign(alignment, size); }static inline void btAlignedFreeDefault(void *ptr) {free(ptr); } #else // 當前編譯環境沒有 對齊的(aligned)內存分配函數 static inline void *btAlignedAllocDefault(size_t size, int alignment) {void *ret;char *real;real = (char *)sAllocFunc(size + sizeof(void *) + (alignment-1)); // 1. 多分配一點內存if (real) {ret = btAlignPointer(real + sizeof(void *),alignment); // 2. 指針調整*((void **)(ret)-1) = (void *)(real); // 3. 登記實際地址} else {ret = (void *)(real);}return (ret); }static inline void btAlignedFreeDefault(void *ptr) {void* real;if (ptr) {real = *((void **)(ptr)-1); // 取出實際內存塊 地址sFreeFunc(real);} } #endif

bullet本身也實現了一個對齊的(aligned)內存分配函數。在系統沒有對齊的內存分配函數的情況下,也能保證btAlignedAllocator::acllocate返回的地址是按特定字節對齊的。

以下就來分析btAlignedAllocDefault / btAlignedFreeDefault是怎樣實現aligned allocation / free的。sAllocFunc/sFreeFunc的定義及初始化:

static void *btAllocDefault(size_t size) {return malloc(size); }static void btFreeDefault(void *ptr) {free(ptr); }static btAllocFunc *sAllocFunc = btAllocDefault; static btFreeFunc *sFreeFunc = btFreeDefault;

bullet同一時候提供了,AllocFunc/FreeFunc的定制化:

void btAlignedAllocSetCustom(btAllocFunc *allocFunc, btFreeFunc *freeFunc) {sAllocFunc = allocFunc ? allocFunc : btAllocDefault;sFreeFunc = freeFunc ? freeFunc : btFreeDefault; } 默認情況下sAllocFunc/sFreeFunc就是malloc/free,btAlignedAllocDefault中可能令人疑惑的是——為什么要多分配一點內存?后面的btAlignPointer有什么用?

再來看看bullet是怎樣實現指針對齊的(btScalar.h):

///align a pointer to the provided alignment, upwards template <typename T>T* btAlignPointer(T* unalignedPtr, size_t alignment) {struct btConvertPointerSizeT{union {T* ptr;size_t integer;};};btConvertPointerSizeT converter;const size_t bit_mask = ~(alignment - 1);converter.ptr = unalignedPtr;converter.integer += alignment-1;converter.integer &= bit_mask;return converter.ptr; }

接下來分析btAlignPointer是怎樣調整指針的?

實際調用btAlignPointer時,使用的alignment都是2的指數。如btAlignedObjectArray使用的是16,以下就以16進行分析。

先如果unalignedPtr是alignment(16)的倍數,則converter.integer += alignment-1; 再 converter.integer &= bit_mask之后,unalignedPtr的值不變。還是alignment(16)的倍數。

再如果unalignedPtr不是alignment(16)的倍數,則converter.integer += alignment-1; 再converter.integer &= bit_mask之后。unalignedPtr的值將被上調到alignment(16)的倍數。

所以btAlignPointer可以將unalignedPtr對齊到alignment倍數。】


明確了btAlignPointer的作用。自然可以明確btAlignedAllocDefault中為什么多申請一點內存,申請的大小是size + sizeof(void *) + (alignment-1):

假設sAllocFunc返回的地址已經依照alignment對齊,則sizeof(void*)和sizeof(alignment-1)及btAlignedAllocDefault的返回值關系例如以下圖所看到的:


void*前面的alignment-sizeof(void*)字節和尾部的sizeof(size)-1字節的內存會被浪費,只是非常小(相對內存條而言)管他呢。


假設sAllocFunc返回的地址沒能按alignment對齊,則sizeof(void*)和sizeof(alignment-1)及btAlignedAllocDefault的返回值關系例如以下圖所看到的:



PS: 順便一提,為什么須要內存對齊?簡單地說。依照機器字長倍數對齊的內存。CPU訪問的速度更快;詳細來說,則要依據詳細CPU和總線控制器的廠商文檔來討論的,那將涉及非常多平臺、硬件細節,所以本文不正確該話題著墨太多。


btAlignedObjectArray——bullet的動態數組

btAlignedObjectArray的作用與STL的vector類似(下面稱std::vector),都是動態數組。btAlignedObjectArray的數據成員(data member)聲明例如以下:

template <typename T> class btAlignedObjectArray {btAlignedAllocator<T , 16> m_allocator; // 沒有data member。不會添加內存int m_size;int m_capacity;T* m_data;//PCK: added this linebool m_ownsMemory; // ... 省略 };

btAlignedObjectArray同一時候封裝了QuickSort。HeapSort。BinarySearch,LinearSearch函數,可用于排序、查找,btAlignedObjectArray的全部成員函數(member function)定義例如以下:

template <typename T> //template <class T> class btAlignedObjectArray {btAlignedAllocator<T , 16> m_allocator;int m_size;int m_capacity;T* m_data;//PCK: added this linebool m_ownsMemory;#ifdef BT_ALLOW_ARRAY_COPY_OPERATOR public:SIMD_FORCE_INLINE btAlignedObjectArray<T>& operator=(const btAlignedObjectArray<T> &other); #else//BT_ALLOW_ARRAY_COPY_OPERATOR private:SIMD_FORCE_INLINE btAlignedObjectArray<T>& operator=(const btAlignedObjectArray<T> &other); #endif//BT_ALLOW_ARRAY_COPY_OPERATORprotected:SIMD_FORCE_INLINE int allocSize(int size);SIMD_FORCE_INLINE void copy(int start,int end, T* dest) const;SIMD_FORCE_INLINE void init();SIMD_FORCE_INLINE void destroy(int first,int last);SIMD_FORCE_INLINE void* allocate(int size);SIMD_FORCE_INLINE void deallocate();public: btAlignedObjectArray();~btAlignedObjectArray();///Generally it is best to avoid using the copy constructor of an btAlignedObjectArray,// and use a (const) reference to the array instead.btAlignedObjectArray(const btAlignedObjectArray& otherArray); /// return the number of elements in the arraySIMD_FORCE_INLINE int size() const;SIMD_FORCE_INLINE const T& at(int n) const;SIMD_FORCE_INLINE T& at(int n);SIMD_FORCE_INLINE const T& operator[](int n) const;SIMD_FORCE_INLINE T& operator[](int n);///clear the array, deallocated memory. Generally it is better to use array.resize(0), // to reduce performance overhead of run-time memory (de)allocations.SIMD_FORCE_INLINE void clear();SIMD_FORCE_INLINE void pop_back();///resize changes the number of elements in the array. If the new size is larger, // the new elements will be constructed using the optional second argument.///when the new number of elements is smaller, the destructor will be called,// but memory will not be freed, to reduce performance overhead of run-time memory (de)allocations.SIMD_FORCE_INLINE void resizeNoInitialize(int newsize);SIMD_FORCE_INLINE void resize(int newsize, const T& fillData=T());SIMD_FORCE_INLINE T& expandNonInitializing( );SIMD_FORCE_INLINE T& expand( const T& fillValue=T());SIMD_FORCE_INLINE void push_back(const T& _Val);/// return the pre-allocated (reserved) elements, this is at least // as large as the total number of elements,see size() and reserve()SIMD_FORCE_INLINE int capacity() const;SIMD_FORCE_INLINE void reserve(int _Count);class less{ public:bool operator() ( const T& a, const T& b ) { return ( a < b ); }};template <typename L>void quickSortInternal(const L& CompareFunc,int lo, int hi);template <typename L>void quickSort(const L& CompareFunc);///heap sort from http://www.csse.monash.edu.au/~lloyd/tildeAlgDS/Sort/Heap/template <typename L>void downHeap(T *pArr, int k, int n, const L& CompareFunc);void swap(int index0,int index1);template <typename L>void heapSort(const L& CompareFunc);///non-recursive binary search, assumes sorted arrayint findBinarySearch(const T& key) const;int findLinearSearch(const T& key) const;void remove(const T& key);//PCK: whole functionvoid initializeFromBuffer(void *buffer, int size, int capacity);void copyFromArray(const btAlignedObjectArray& otherArray); };

btAlignedObjectArray和std::vector類似。各成員函數的詳細實現這里不再列出。


std::unordered_map的內存布局

btHashMap的內存布局與我們常見的HashMap的內存布局截然不同。為了和btHashMap的內存布局對照,這里先介紹一下std::unordered_map的內存布局。

GCC中std::unordered_map僅是對_Hahstable的簡單包裝。_Hashtable的數據成員定義例如以下:

__bucket_type* _M_buckets;size_type _M_bucket_count;__before_begin _M_bbegin;size_type _M_element_count;_RehashPolicy _M_rehash_policy;當中。size_type為std::size_t的typedef;而_RehashPlolicy是詳細的策略類,僅僅有成員函數定義,沒有數據成員(這是一種被稱作Policy Based的設計范式。詳細可參閱《Modern C++ Design》,中譯本名為《C++設計新思維》。由侯捷先生翻譯)。

繼續跟蹤_bucket_type,能夠看到(_Hashtable):

using __bucket_type = typename __hashtable_base::__bucket_type;和(__hashtable_base):

using __node_base = __detail::_Hash_node_base;using __bucket_type = __node_base*;

至此。才知道_M_buckets的類型為:_Hash_node_base**

繼續追蹤。能夠看到_Hash_node_base的定義: /*** struct _Hash_node_base** Nodes, used to wrap elements stored in the hash table. A policy* template parameter of class template _Hashtable controls whether* nodes also store a hash code. In some cases (e.g. strings) this* may be a performance win.*/struct _Hash_node_base{_Hash_node_base* _M_nxt;_Hash_node_base() : _M_nxt() { }_Hash_node_base(_Hash_node_base* __next) : _M_nxt(__next) { }};

從_Hashtable::_M_buckets(二維指針)和_Hash_node_base的_M_nxt的類型(指針)。能夠推測Hashtable的內存布局——buckets數組存放hash值同樣的node鏈表的頭指針,每一個bucket上掛著一個鏈表。

繼續看__before_begin的類型(_Hashtable):

using __before_begin = __detail::_Before_begin<_Node_allocator_type>;繼續跟蹤: /*** This type is to combine a _Hash_node_base instance with an allocator* instance through inheritance to benefit from EBO when possible.*/template<typename _NodeAlloc>struct _Before_begin : public _NodeAlloc{_Hash_node_base _M_node;_Before_begin(const _Before_begin&) = default;_Before_begin(_Before_begin&&) = default;template<typename _Alloc>_Before_begin(_Alloc&& __a): _NodeAlloc(std::forward<_Alloc>(__a)){ }};依據對STL雙鏈表std::list的了解,能夠推測Berfore_begin的作用,非常可能和雙鏈表的“頭部的多余的一個節點”類似,僅僅是為了方便迭代器(iterator)迭代,通過_Hashtable::begin()能夠得到驗證: iteratorbegin() noexcept{ return iterator(_M_begin()); }__node_type*_M_begin() const{ return static_cast<__node_type*>(_M_before_begin()._M_nxt); }const __node_base&_M_before_begin() const{ return _M_bbegin._M_node; }


實際存放Value的node類型為以下兩種的當中一種(按Hash_node_base的凝視,Key為string時可能會用第一種,以提升性能):

/*** Specialization for nodes with caches, struct _Hash_node.** Base class is __detail::_Hash_node_base.*/template<typename _Value>struct _Hash_node<_Value, true> : _Hash_node_base{_Value _M_v;std::size_t _M_hash_code;template<typename... _Args>_Hash_node(_Args&&... __args): _M_v(std::forward<_Args>(__args)...), _M_hash_code() { }_Hash_node*_M_next() const { return static_cast<_Hash_node*>(_M_nxt); }};/*** Specialization for nodes without caches, struct _Hash_node.** Base class is __detail::_Hash_node_base.*/template<typename _Value>struct _Hash_node<_Value, false> : _Hash_node_base{_Value _M_v;template<typename... _Args>_Hash_node(_Args&&... __args): _M_v(std::forward<_Args>(__args)...) { }_Hash_node*_M_next() const { return static_cast<_Hash_node*>(_M_nxt); }};

以下通過insert源代碼的追蹤,證實我們對hashtable內存布局的猜想:

_Hashtable::insert:

template<typename _Pair, typename = _IFconsp<_Pair>>__ireturn_typeinsert(_Pair&& __v){__hashtable& __h = this->_M_conjure_hashtable();return __h._M_emplace(__unique_keys(), std::forward<_Pair>(__v));}

_Hashtable::_M_emplace(返回值類型寫得太復雜,已刪除):

_M_emplace(std::true_type, _Args&&... __args){// First build the node to get access to the hash code__node_type* __node = _M_allocate_node(std::forward<_Args>(__args)...); // 申請鏈表節點 __args為 pair<Key, Value> 類型const key_type& __k = this->_M_extract()(__node->_M_v); // 從節點中抽取 key__hash_code __code; __try{__code = this->_M_hash_code(__k);}__catch(...){_M_deallocate_node(__node);__throw_exception_again;}size_type __bkt = _M_bucket_index(__k, __code); // 尋找buckets上的相應hash code相應的indexif (__node_type* __p = _M_find_node(__bkt, __k, __code)) // 在bucket所指鏈表上找到實際節點{// There is already an equivalent node, no insertion_M_deallocate_node(__node);return std::make_pair(iterator(__p), false);}// Insert the nodereturn std::make_pair(_M_insert_unique_node(__bkt, __code, __node),true);}
_Hashtable::_M_find_node: __node_type*_M_find_node(size_type __bkt, const key_type& __key,__hash_code __c) const{__node_base* __before_n = _M_find_before_node(__bkt, __key, __c);if (__before_n)return static_cast<__node_type*>(__before_n->_M_nxt);return nullptr;}
_Hashtable::_M_find_before_node(返回值類型寫得太復雜,已刪除): _M_find_before_node(size_type __n, const key_type& __k,__hash_code __code) const{__node_base* __prev_p = _M_buckets[__n]; // 取出頭指針if (!__prev_p)return nullptr;__node_type* __p = static_cast<__node_type*>(__prev_p->_M_nxt);for (;; __p = __p->_M_next()) // 遍歷鏈表{if (this->_M_equals(__k, __code, __p)) // key匹配?return __prev_p;if (!__p->_M_nxt || _M_bucket_index(__p->_M_next()) != __n)break;__prev_p = __p;}return nullptr;}

看到_Hashtable::_M_find_before_node的代碼,就驗證了此前我們對于Hashtable內存布局的猜想:這和SGI hash_map的實現體hashtable的內存布局同樣(詳情可參考《STL源代碼剖析》,侯捷先生著)。

(PS:追蹤起來并不輕松,能夠借助Eclipse等集成開發環境進行)

比如,std::unordered_map<int, int*>背后的Hashtable的一種可能的內存布局例如以下:


std::unordered_map的內存布局是大多數<數據結構>、<算法>類教材給出的“標準做法”,也是比較常見的實現方法。


btHashMap

btHashMap的內存布局。與“標準做法”截然不同。例如以下可見btHashMap的數據成員(data member)定義:

template <class Key, class Value> class btHashMap {protected:btAlignedObjectArray<int> m_hashTable;btAlignedObjectArray<int> m_next;btAlignedObjectArray<Value> m_valueArray;btAlignedObjectArray<Key> m_keyArray; // ... 省略 };能夠看到,btHashMap的將buckets和key, value全放在一起。它的內存布局可能例如以下:

依據命名。能夠推測:

m_keyArray和m_valueArray分別存放key和value;

m_next的作用應該是存放k/v array的index,以此形成鏈表。

m_hashTable的作用應該和前面_M_bukets所指向的數組類似,作為表頭;

以下通過分析btHashMap的幾個方法,來對這幾個推測一一驗證。


btHashMap::findIndex

以下來看看btHashMap::findIndex的實現:

int findIndex(const Key& key) const{unsigned int hash = key.getHash() & (m_valueArray.capacity()-1); // 依賴 Key::getHash()if (hash >= (unsigned int)m_hashTable.size()){return BT_HASH_NULL;}int index = m_hashTable[hash]; // index相當于unordered_map的buckets[hash]的鏈表頭指針while ((index != BT_HASH_NULL) && key.equals(m_keyArray[index]) == false) // 遍歷鏈表。直到匹配,依賴 Key::equals(Key){index = m_next[index]; }return index;}

btHashMap::findIndex用到了m_hashTable。m_keyArray,m_next,能夠看出:

m_hashTable的作用確實類似于unordered_map的buckets數組;

m_keyArray確實是存放了key;

m_next[i]確實類似于unordered_map鏈表節點的next指針。


btHashMap::insert

接下來看看btHashMap::insert:

void insert(const Key& key, const Value& value) {int hash = key.getHash() & (m_valueArray.capacity()-1);//replace value if the key is already thereint index = findIndex(key); // 找到了<Key, Value>節點if (index != BT_HASH_NULL){m_valueArray[index]=value; // 找到了,更行valuereturn;}int count = m_valueArray.size(); // 當前已填充數目int oldCapacity = m_valueArray.capacity();m_valueArray.push_back(value); // value壓入m_valueArray的尾部,capacity可能增長m_keyArray.push_back(key); // key壓入m_keyArray的尾部int newCapacity = m_valueArray.capacity();if (oldCapacity < newCapacity) {growTables(key); // 假設增長。調整其余兩個數組的大小。并調整頭指針所在位置//hash with new capacityhash = key.getHash() & (m_valueArray.capacity()-1);}m_next[count] = m_hashTable[hash]; // 連同下一行,將新節點插入 m_hashTable[hash]鏈表頭部m_hashTable[hash] = count;}

這里驗證了:m_valueArray存放的確實是value。


btHashMap::remove

btHashMap與普通Hash表的差別在于,它可能要自己管理節點內存;比方,中間節點remove掉之后。怎樣保證下次insert可以復用節點內存?通過btHashMap::remove可以知道bullet是怎樣實現的:

void remove(const Key& key) {int hash = key.getHash() & (m_valueArray.capacity()-1);int pairIndex = findIndex(key); // 找到<Key, Value>的 indexif (pairIndex ==BT_HASH_NULL){return;}// Remove the pair from the hash table.int index = m_hashTable[hash]; // 取出頭指針btAssert(index != BT_HASH_NULL);int previous = BT_HASH_NULL;while (index != pairIndex) // 找index的前驅{previous = index;index = m_next[index];}if (previous != BT_HASH_NULL) // 將當前節點從鏈表上刪除{btAssert(m_next[previous] == pairIndex);m_next[previous] = m_next[pairIndex]; // 當前節點位于鏈表中間}else {m_hashTable[hash] = m_next[pairIndex]; // 當前節點是鏈表第一個節點}// We now move the last pair into spot of the// pair being removed. We need to fix the hash// table indices to support the move.int lastPairIndex = m_valueArray.size() - 1; // If the removed pair is the last pair, we are done.if (lastPairIndex == pairIndex) // 假設<Key, Value>已經是array的最后一個元素。則pop_back將減小size(capacity不變){m_valueArray.pop_back();m_keyArray.pop_back();return;}// Remove the last pair from the hash table. 將最后一個<Key, Value>對從array上移除int lastHash = m_keyArray[lastPairIndex].getHash() & (m_valueArray.capacity()-1);index = m_hashTable[lastHash];btAssert(index != BT_HASH_NULL);previous = BT_HASH_NULL;while (index != lastPairIndex){previous = index;index = m_next[index];}if (previous != BT_HASH_NULL){btAssert(m_next[previous] == lastPairIndex);m_next[previous] = m_next[lastPairIndex];}else{m_hashTable[lastHash] = m_next[lastPairIndex];}// Copy the last pair into the remove pair's spot. 將最后一個<Key, Value>復制到移除pair的空當處m_valueArray[pairIndex] = m_valueArray[lastPairIndex];m_keyArray[pairIndex] = m_keyArray[lastPairIndex];// Insert the last pair into the hash table , 將移除節點插入到m_hashTable[lastHash]鏈表的頭部m_next[pairIndex] = m_hashTable[lastHash];m_hashTable[lastHash] = pairIndex;m_valueArray.pop_back();m_keyArray.pop_back();}

內存緊密(連續)的優點

btHashMap的這樣的設計。可以保證整個Hash表內存的緊密(連續)性。而這樣的連續性的優點主要在于:

第一,能與數組(指針)式API兼容。比方非常多OpenGL API。由于存在btHashMap內的Value和Key在內存上都是連續的。所以這一點非常好理解;

第二,保證了cache命中率(表元素較少時)。因為普通鏈表的節點內存是在每次須要時才申請的,所以基本上不會連續。通常不在同樣內存頁。所以,即便是短時間內多次訪問鏈表節點,也可能因為節點內存分散造成不能將全部節點放入cache。從而導致訪問速度的下降;而btHashMap的節點內存始終連續,因而保證較高的cache命中率,能帶來一定程度的性能提升。


btAlignedAllocator點評

btAlignedAllocator定制化接口與std::allocator全然不同。std::allocator的思路是:首先實現allocator,然后將allocator作為模板參數寫入詳細數據結構上,如vector<int, allocator<int> >;

這樣的方法盡管能夠實現“定制化”,但存在著一定的問題:

第一,因為全部標準庫的allcoator用的都是std::allocator。假設你使用了第二種allocator,程序中就可能存在不止一種類型的內存管理方法一起工作的局面;特別是當標準庫使用的是SGI 當年實現的“程序退出時才歸還全部內存的”allocator(詳細可參閱《STL源代碼剖析》)時,內存爭用是不可避免的。

第二。這樣的設計無形中添加了編碼和調試的復雜性。相信調試過gcc STL代碼的人深有體會。

而btAlignedAllocator則全然不存在這種問題:

第一。它的allocate/deallocate行為通過全局的函數指針代理實現,不可能存在同一時候有兩個以上的類型底層管理內存的方法。

第二。使用btAlignedAllocator的數據結構,其模板參數相對簡單。編碼、調試的復雜性自然也減少了。

本人拙見,STL有點過度設計了。盡管Policy Based的設計可以帶來靈活性,但代碼的可讀性下降了非常多(也許開發glibc++的那群人沒打算讓別人看他們的代碼?)。


擴展閱讀

文中提到了兩本書:

《Modern C++ Design》(中譯本名為《C++設計新思維》。侯捷先生譯)。該書仔細描寫敘述了Policy Based Design。

《STL源代碼剖析》(侯捷先生著)。該書具體剖析了SGI hashtable的實現。

本文所討論的源代碼版本號:

bullet 2.81

gcc 4.6.1(MinGW)


歡迎評論或email(xusiwei1236@163.com)交流觀點,轉載注明出處。勿做商用。

轉載于:https://www.cnblogs.com/yxwkf/p/5276686.html

總結

以上是生活随笔為你收集整理的bullet HashMap 内存紧密的哈希表的全部內容,希望文章能夠幫你解決所遇到的問題。

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

91超碰在线播放 | 久日视频| 国产一二三区在线观看 | 欧美午夜精品久久久久 | 在线观看黄色小视频 | 久久天天躁夜夜躁狠狠85麻豆 | 亚洲精品a区 | 日韩高清在线看 | 亚洲精品一区二区在线观看 | 色94色欧美 | 精品一区二区三区在线播放 | 99国产精品免费网站 | 99久久99视频只有精品 | 成人av在线影视 | 免费国产一区二区视频 | www好男人| 欧美日韩一区二区三区不卡 | www.99av | 看片的网址 | 国产精品久久久久久久久久妇女 | av怡红院 | 香蕉视频在线播放 | 久久久精品国产免费观看一区二区 | 午夜av在线电影 | 2023av在线 | 综合久久久久 | 天天操天天操天天操 | 欧美人体xx | 久久综合9988久久爱 | 免费能看的av | 在线免费观看黄网站 | av成人动漫 | 99热在线观看| 久久免费看a级毛毛片 | 天天摸天天操天天爽 | 成人国产精品av | 麻豆视频免费 | 色综合五月天 | 91精品视频免费观看 | 国产精品二区在线观看 | 国产999精品视频 | 久久综合久久综合这里只有精品 | 在线观看国产一区二区 | 欧美最新大片在线看 | 日韩在线视频网站 | 久久久久区 | 激情电影影院 | 射射射av | 婷婷丁香在线视频 | 毛片无卡免费无播放器 | 成片视频在线观看 | 久久免费视频网 | 91网免费看 | 美女黄频在线观看 | 黄色一及电影 | 一区在线播放 | 激情开心站| 国产精品99久久久久久有的能看 | 麻豆 videos | 天天射天天色天天干 | www.在线观看av | 免费视频色| 91精品久久久久久粉嫩 | 在线观看你懂的网址 | 免费观看9x视频网站在线观看 | 久久久久久国产精品 | 91福利视频网站 | 欧美大片在线观看一区 | 在线观看视频h | 亚洲国产成人精品在线观看 | 亚洲高清视频一区二区三区 | 天天搞天天 | 91色蜜桃 | 精品久久久999| 91麻豆国产福利在线观看 | 久久成人国产精品免费软件 | 精品亚洲视频在线 | 中文字幕在线视频免费播放 | 国产99在线免费 | 日韩精品一区二区免费视频 | 超碰在线98| 操操操夜夜操 | 91麻豆传媒 | 国内99视频 | 日韩欧美在线观看一区二区 | 伊人久久国产精品 | 激情五月婷婷 | 久久9999久久免费精品国产 | 99久久精品久久久久久动态片 | 97碰碰视频 | 日韩一区二区免费播放 | 粉嫩高清一区二区三区 | 日韩欧美电影网 | 成人a大片 | a久久久久久 | 美女国产精品 | 国内精品久久久久久久影视简单 | 成年人网站免费观看 | 亚洲精品五月天 | 国产一区二区在线免费视频 | 黄色三级网站 | 国产精品手机看片 | 免费在线国产黄色 | 天天操天天谢 | 亚洲精品高清一区二区三区四区 | av超碰在线观看 | www.天天操 | 国产99一区二区 | 麻豆91精品视频 | 521色香蕉网站在线观看 | 五月婷婷丁香激情 | 伊人看片 | 九九九热精品 | 黄色国产成人 | 精品久久一 | 91精品视频在线免费观看 | 日韩黄色免费在线观看 | 毛片视频网址 | 天天想夜夜操 | 亚洲视频综合 | 五月婷婷综合激情网 | 成人午夜电影久久影院 | 91麻豆文化传媒在线观看 | 欧美国产三区 | 99久久精品国产毛片 | 亚洲精品乱码久久久久久蜜桃动漫 | 中文国产字幕 | 日本精品在线 | 亚洲国产日本 | 日韩城人在线 | 久久久久久毛片精品免费不卡 | 久久国产精品99久久久久 | 91免费视频黄| 欧美另类视频 | 国产九色在线播放九色 | 在线免费试看 | 91av免费在线观看 | 一区二区精品在线 | 精品国产免费人成在线观看 | 日韩在线一级 | 四虎永久免费在线观看 | 精品久久久久久综合日本 | 91成人精品视频 | av福利在线播放 | 黄色大全视频 | 久久精品久久精品久久精品 | 91精品免费 | 五月婷在线观看 | 国产xvideos免费视频播放 | 国产无区一区二区三麻豆 | 91成人精品一区在线播放69 | 91免费版成人 | 成年美女黄网站色大片免费看 | 人人干在线 | 88av视频 | 毛片a级片 | 久久99国产精品免费 | 久久久久久久久久免费视频 | 国产精品99在线播放 | 九九热视频在线免费观看 | 五月激情视频 | 国际精品久久久久 | 免费在线观看视频a | 日韩成人精品 | 国产又粗又猛又色又黄视频 | 精品欧美一区二区精品久久 | 国产成人777777 | 久久综合久久综合久久综合 | 综合网伊人 | 久久久免费精品国产一区二区 | 91精品秘密在线观看 | 天堂av网站 | 国产精品毛片一区视频播 | 国产一区二区三区高清播放 | 在线97| 91桃色国产在线播放 | 国产伦精品一区二区三区照片91 | 999国产精品视频 | 丝袜美女在线观看 | 天天做综合网 | 欧美日本一二三 | 国产精品6| av免费在线观看1 | 91传媒视频在线观看 | 91精品免费在线观看 | 久久综合婷婷国产二区高清 | 国产精品video爽爽爽爽 | 久久中文字幕视频 | 超碰免费97 | 国产一区观看 | 国产高清视频在线观看 | 中文字幕精品三级久久久 | 日韩在线视频一区 | 久久精彩免费视频 | 91九色视频 | 69av久久| 最近日本mv字幕免费观看 | 日本最新高清不卡中文字幕 | 久久 地址 | 天天射天天干天天 | 粉嫩av一区二区三区免费 | 午夜av在线电影 | 中文字幕一区二区三区四区 | 日韩av片在线 | 国内精品久久久久久久影视麻豆 | 免费中午字幕无吗 | 亚洲精品中文字幕视频 | 激情黄色av | 精品国精品自拍自在线 | 西西www444| 久久夜色电影 | 手机av在线不卡 | 日韩在线视频线视频免费网站 | 国产人成精品一区二区三 | 91成人欧美 | 久久综合中文色婷婷 | av一区二区三区在线播放 | 最近中文字幕在线播放 | 亚洲黄色免费在线 | 天天插天天干 | 国产高清久久久久 | 免费看一及片 | 免费看黄色小说的网站 | 日韩在线视频线视频免费网站 | 狠狠狠狠狠狠干 | 不卡的av在线播放 | 又黄又爽又无遮挡免费的网站 | 久久免费播放 | 婷婷在线色 | 成人宗合网 | 亚洲精品乱码久久久久久高潮 | 激情综合五月网 | 日韩在线观看视频中文字幕 | 99久久精品免费视频 | 亚洲另类视频 | 狠狠色丁香婷婷综合 | 日本久久中文 | 精品视频123区在线观看 | av观看网站 | 五月婷婷久久综合 | 福利视频一区二区 | 人人干97 | 99久久精 | 国产精品一区二区你懂的 | 日韩美精品视频 | 国产91在线观 | 91久久影院 | 国产麻豆电影在线观看 | 国产一区在线免费观看视频 | 亚洲精品大片www | 五月婷婷一区 | 欧美日韩高清免费 | 高清日韩一区二区 | 国产精品午夜8888 | 91爱爱电影 | 国产精品久久久久aaaa | 国产高清黄| 久久精品老司机 | 亚洲午夜在线视频 | 日韩欧美在线观看 | 亚洲激情一区二区三区 | 9999激情| 精品色999| 亚洲理论电影网 | 亚洲色图22p | 91精品啪在线观看国产线免费 | 9999在线| 欧美a在线免费观看 | 91看片麻豆 | 日韩高清成人 | 五月激情姐姐 | 国产97在线视频 | 欧美成人精品欧美一级乱黄 | 久久中文精品视频 | 91av在| 日本公妇色中文字幕 | 国产黄色免费看 | www.成人精品 | 黄色免费网战 | 久久人91精品久久久久久不卡 | 亚洲精品高清在线观看 | 久久午夜电影院 | 日日操日日插 | 六月婷色| 亚洲精品久久久久久中文传媒 | 国产精品系列在线播放 | 超碰97中文 | 欧美黄在线 | 成人影视免费看 | 日韩r级电影在线观看 | 麻豆91精品视频 | 国产黄色av影视 | 久久一区二区三区国产精品 | 99久久精品国产一区二区三区 | 天天色天天干天天 | 日韩三级在线观看 | 国产美女视频免费观看的网站 | 日韩在线观看一区二区 | 国产在线一线 | 亚洲美女精品区人人人人 | 99久久这里只有精品 | 一区二区三区免费在线观看视频 | 麻豆视频免费在线观看 | 欧美日韩视频精品 | 伊人日日干| 日韩二区在线观看 | 制服丝袜天堂 | 欧洲av在线 | 久久亚洲综合国产精品99麻豆的功能介绍 | 久久久久久毛片精品免费不卡 | www.亚洲激情.com | 欧美午夜性生活 | 国产96在线观看 | 久草在线99 | 免费看的黄色的网站 | 欧美精品免费在线 | 91污污| 久操视频在线 | 亚洲理论视频 | 五月婷婷狠狠 | 国产精品免费视频观看 | 五月天久久激情 | 免费在线播放黄色 | 国产精品免费观看国产网曝瓜 | 热久久最新地址 | 国产999精品久久久久久麻豆 | 亚洲区另类春色综合小说校园片 | 日韩在线观看中文 | 午夜av电影 | 日韩有色 | 亚洲爱爱视频 | 久久精品网站免费观看 | 在线99| 成人毛片在线观看 | 成人四虎 | 免费高清在线视频一区· | 日本中文字幕影院 | 精品福利在线 | 日本黄色免费观看 | 黄色福利视频网站 | 开心婷婷色 | 六月激情婷婷 | 91x色 | 精品国产乱子伦一区二区 | 成人影视免费 | 亚洲国产美女精品久久久久∴ | 欧美激情视频一区 | 国语精品久久 | 毛片网站在线 | 黄色日视频 | 97超碰国产精品女人人人爽 | 国产精品久久久久久久久久久久午夜 | 日本xxxx裸体xxxx17 | 最近中文字幕高清字幕在线视频 | 免费a v在线| 亚洲成人精品av | 免费在线观看av网址 | 亚洲一区二区三区四区在线视频 | 色天堂在线视频 | 胖bbbb搡bbbb擦bbbb | 久久精品亚洲一区二区三区观看模式 | 国产理论一区二区三区 | 免费国产ww | 日日天天av | 激情视频综合网 | 69国产在线观看 | 久草精品在线 | 亚洲精品在线视频网站 | 天天搞天天干 | 玖玖视频免费在线 | 亚洲五月 | 欧美肥妇free | 久久精美视频 | 99热精品视 | av在线之家电影网站 | 青草视频在线免费 | 2024国产精品视频 | 国产精品第一页在线 | 91香蕉嫩草 | 97人人看 | 国产成人99av超碰超爽 | 久久影院亚洲 | 国产一区二区高清不卡 | 国产手机在线视频 | 久久在线免费视频 | 天天插天天射 | 4438全国亚洲精品在线观看视频 | 亚洲男男gaygayxxxgv | 国产精品网红直播 | www.在线看片.com| 91精品国自产拍天天拍 | 国产一级片不卡 | 99中文在线 | 婷婷狠狠操 | 日韩试看 | 91精品国产99久久久久久红楼 | 国内精品免费久久影院 | 日韩中文字幕a | www.国产在线视频 | 亚洲精品人人 | 精品国产免费看 | 成人a毛片 | 美女视频黄网站 | 美女网站一区 | 日韩电影在线看 | 久久国产免 | 精品极品在线 | 涩涩在线 | av综合站| 中文字幕免费观看 | 国产97在线看 | 天天干天天综合 | 久久天天拍 | 在线观看黄色国产 | 丁香婷婷久久久综合精品国产 | 婷婷久久婷婷 | 九色最新网址 | 波多野结衣最新 | 伊人婷婷网 | av网站在线观看播放 | 日韩av成人在线 | 国产不卡视频在线播放 | 久久久久免费 | 97精品国产| 国产亚洲精品成人av久久ww | 九九热在线视频 | 亚洲 中文 欧美 日韩vr 在线 | 国产人成一区二区三区影院 | 久草香蕉在线 | 婷婷综合久久 | 五月天免费网站 | 成人免费视频在线观看 | 色狠狠狠 | 在线视频久 | 久久久久久综合 | 成人av电影在线播放 | 亚洲最新在线视频 | 久久综合婷婷国产二区高清 | 色婷婷亚洲婷婷 | 亚洲第一区在线播放 | 国产成人精品久 | 精品国产一区在线观看 | 国产免费又爽又刺激在线观看 | 欧美日韩视频在线播放 | 久久伦理电影 | 操操综合| 色综合久久久久综合体桃花网 | 亚洲日韩中文字幕 | 中文字幕精品视频 | 五月婷婷六月综合 | 99久久精品国产免费看不卡 | 在线亚洲欧美日韩 | 一区二区三区 中文字幕 | 激情 亚洲| 丁香六月av | 欧美a级一区二区 | 中文十次啦 | 久草免费福利在线观看 | 国产视频一区在线 | 欧美精品一区二区在线播放 | 国产亚洲午夜高清国产拍精品 | 97热在线观看 | 一级片免费观看视频 | 国产黑丝一区二区 | 亚洲 欧美 变态 国产 另类 | 免费高清男女打扑克视频 | 国产又粗又硬又长又爽的视频 | 精品国产一区二区三区在线观看 | 国产视频一区二区在线 | 久久久久国产精品免费 | www黄色软件| 一区二区三区不卡在线 | 久久久香蕉视频 | 亚洲日韩欧美视频 | 国产精品久久久久久久久免费 | 婷婷在线精品视频 | 久久女教师 | 麻豆精品在线视频 | 欧美精品国产精品 | 久久精品爱爱视频 | 免费看色网站 | 高清视频一区 | 日本高清中文字幕有码在线 | 天无日天天操天天干 | 欧美日韩精品区 | 亚洲视屏在线播放 | 在线观看日本高清mv视频 | 亚洲最新精品 | 成人免费 在线播放 | 国产一区二区三区 在线 | 国产精品永久在线 | 91桃色在线免费观看 | 久久久久福利视频 | 999日韩 | 九九国产精品视频 | 亚洲成aⅴ人在线观看 | 特片网久久 | 久久精品99国产精品 | 亚洲国产精品视频在线观看 | 黄色a视频免费 | 国产亚洲综合精品 | 国产精品va在线观看入 | 亚洲区另类春色综合小说校园片 | 国产又黄又爽又猛视频日本 | 97超碰资源 | 久久精视频 | 一区二区三区不卡在线 | 三上悠亚一区二区在线观看 | 99综合视频 | 在线v片免费观看视频 | 久久免费精品视频 | 精品一区二三区 | 91麻豆精品国产91久久久无需广告 | 久久韩国免费视频 | 午夜国产一区二区三区四区 | 国产精品系列在线 | 91免费看片黄 | 色婷婷亚洲婷婷 | 亚洲成a人片综合在线 | 又湿又紧又大又爽a视频国产 | 操操操天天操 | 欧美激情综合色综合啪啪五月 | 久久久精品成人 | 国产精品成人av电影 | 久久毛片高清国产 | 欧美激情视频一二三区 | 黄色在线免费观看网站 | 久久综合九色综合97_ 久久久 | 国产精品理论在线观看 | 在线国产能看的 | 91传媒在线| 欧美色图一区 | 亚洲一区视频免费观看 | 亚洲国产精品视频在线观看 | 香蕉成人在线视频 | 国产日产精品久久久久快鸭 | 日日夜夜免费精品 | 免费在线观看国产精品 | 国产一区二区三区 在线 | 国产成人高清在线 | 亚洲区精品 | 国产黄免费 | 欧美日韩一区二区视频在线观看 | 97色婷婷人人爽人人 | 国产精品门事件 | 天天操天天射天天插 | 天天干天天操天天射 | 亚州国产精品视频 | 国产视频不卡 | 亚洲国产操 | 国产护士av | 午夜电影av | 日韩激情一二三区 | 中文字幕在线影视资源 | 天天色天天搞 | 久久在线观看视频 | 97在线超碰 | www.久久99 | 国产一级片在线播放 | 免费看黄色小说的网站 | 三级免费黄色 | 久久国产欧美日韩 | 9999精品| 91九色视频 | 超碰免费av | 伊人久久婷婷 | 欧美日韩国产一二三区 | 午夜久久久久久久久久久 | 九九九热精品免费视频观看 | 国产美女视频免费观看的网站 | 福利视频精品 | 国产成人精品福利 | 久久精品香蕉视频 | 国产精品久久电影观看 | 亚洲伊人第一页 | 久久久人人人 | 黄色av观看| 99热.com| 久久与婷婷 | 婷婷色综合 | 娇妻呻吟一区二区三区 | 日韩电影一区二区在线 | 99精品成人 | 亚洲禁18久人片 | 国产精品福利无圣光在线一区 | 国产91小视频 | 久久高清免费 | 亚洲精品在线视频观看 | 久久99网站| 久久国产精品一二三区 | 丰满少妇在线观看网站 | 中文字幕一区二区三区四区 | 成人aaa毛片 | 在线观看视频在线观看 | 亚洲va综合va国产va中文 | 国产一级一片免费播放放a 一区二区三区国产欧美 | 亚洲综合欧美日韩狠狠色 | 久久国产精品99久久人人澡 | 亚洲国产欧美在线看片xxoo | 九九免费在线观看视频 | 99久久婷婷国产综合精品 | 免费的成人av | 日日射av | 人人看97 | 中文字幕最新精品 | 在线日本看片免费人成视久网 | 337p日本欧洲亚洲大胆裸体艺术 | 国产视频资源 | 黄色大片免费网站 | 亚洲一级电影在线观看 | 久久精品欧美一区二区三区麻豆 | 久操视频在线免费看 | 婷婷国产v亚洲v欧美久久 | 国产福利一区在线观看 | 亚洲a在线观看 | 久久成人在线视频 | 国产成人99av超碰超爽 | 久草电影在线 | 免费日韩av片 | 国产二区视频在线观看 | 亚洲成人精品在线 | 中文字幕乱码日本亚洲一区二区 | 久久综合久久综合九色 | 久久激情精品 | 日韩99热| 日本久久精 | 免费三级a| 日本成人免费在线观看 | 国产精品黑丝在线观看 | 精品久久久久久亚洲综合网站 | 97在线观视频免费观看 | 伊人久久av | 91c网站色版视频 | 黄色小视频在线观看免费 | 丁香六月婷婷激情 | 日韩影片在线观看 | 国产视频一区二区在线 | 国产成人高清 | 久久精品99久久久久久2456 | 91在线免费播放 | 亚洲黄色大片 | 在线观看黄| 久久综合亚洲鲁鲁五月久久 | 日韩欧美在线观看 | 精品视频免费观看 | 亚洲精品av中文字幕在线在线 | 日韩欧美在线观看 | 免费人成在线观看网站 | 国产成人一级电影 | 久久理伦片 | 成年人视频在线观看免费 | 色婷婷激情四射 | 日韩区欧美久久久无人区 | av高清网站在线观看 | 日韩经典一区二区三区 | 国产精品久久久一区二区 | 久久成人毛片 | 91精品国产一区 | 久久久精品福利视频 | 国产高清专区 | 麻豆91在线观看 | 欧美a级在线免费观看 | 免费看污在线观看 | bbb搡bbb爽爽爽 | 三级av网| 91人人澡人人爽 | 亚洲第一中文网 | 免费观看性生活大片3 | 在线欧美国产 | 久久精品视频在线免费观看 | 久久人人97超碰国产公开结果 | 亚洲精品成人网 | 色5月婷婷 | 青青草国产在线 | 在线播放 日韩专区 | 天天色天天综合 | 久久成人高清 | 中文字幕在线人 | 亚洲播播| 日韩欧美精品在线观看视频 | 91看片网址 | 欧美成人在线免费观看 | 香蕉视频在线免费看 | 在线观看岛国 | 久久久一本精品99久久精品 | 黄色aaa毛片| 亚洲激情一区二区三区 | 亚洲国产精品va在线看黑人 | 亚洲精品 在线视频 | 国产一区不卡在线 | 国产理论片在线观看 | 欧美性大胆 | 日韩欧美在线综合网 | 精品专区一区二区 | 欧美视频99| 欧美一区在线观看视频 | 中文字幕资源网在线观看 | 亚洲天堂香蕉 | 国产精品视频 | 亚洲精品国产精品国自 | 欧美日韩高清一区二区 国产亚洲免费看 | 国产做aⅴ在线视频播放 | 在线观看91网站 | 五月婷婷在线视频观看 | 综合黄色网 | 日韩mv欧美mv国产精品 | 超级碰碰碰视频 | 亚洲欧美成人网 | 在线观看日韩一区 | 中文av在线天堂 | 日韩中文字幕免费在线观看 | 国产成人综合图片 | 亚洲视频在线观看网站 | 玖草影院 | av在线播放国产 | 久久精品国产精品亚洲精品 | 国产操在线 | 西西4444www大胆艺术 | 国产黄色av网站 | 欧美片一区二区三区 | 欧美在线视频a | 日韩黄色大片在线观看 | 蜜桃麻豆www久久囤产精品 | 天天干婷婷| 成人久久18免费网站图片 | 天天激情综合网 | 中文字幕在线资源 | 国产黄色精品在线观看 | 女人18毛片a级毛片一区二区 | 国产精品成人一区二区三区吃奶 | 国产午夜精品久久 | 国产成人精品区 | 91免费网站在线观看 | 美女视频黄的免费的 | 国产人成在线观看 | www.99热精品| 国产精品 亚洲精品 | 看污网站 | 天天综合网国产 | 精品91| 天天摸天天弄 | 人人看黄色 | 97超在线| 人人干天天干 | 亚洲精品国产高清 | 国产精品成人av电影 | 午夜视频日本 | 久久精品永久免费 | 欧美午夜理伦三级在线观看 | av在线一级 | 久久免费视频播放 | 亚洲一区二区精品视频 | 久久精品视频3 | 九九九九热精品免费视频点播观看 | 日韩欧美视频免费在线观看 | 亚洲一区精品人人爽人人躁 | 五月激情丁香婷婷 | 成 人 黄 色 视频免费播放 | 在线 视频 一区二区 | 久久久免费少妇 | 亚洲狠狠操 | 日韩在线中文字幕视频 | 91在线入口 | 免费97视频 | 夜夜操天天摸 | 久草在线中文888 | 国产精品第一页在线观看 | 丝袜美女视频网站 | 精品国产一区二区三区四 | 久草在线观 | 日韩小视频 | 欧美日韩精品久久久 | 日本大片免费观看在线 | 深爱激情久久 | 久久久久国产成人免费精品免费 | 免费看一及片 | 国产视频一区在线免费观看 | 97视频人人| 精品成人a区在线观看 | 四虎影视av | 亚洲天堂网在线播放 | 欧美另类tv| av资源中文字幕 | a级一a一级在线观看 | 国产在线精品一区二区不卡了 | 91丨九色丨蝌蚪丨对白 | 久久公开免费视频 | 国产精品免费av | 国产在线观看免费 | 成 人 黄 色视频免费播放 | 亚洲二区精品 | 中文字幕在线观看播放 | 2019久久精品 | 亚洲视频精品在线 | 国产成人精品日本亚洲999 | 久久久精品国产免费观看同学 | 国产一级黄色片免费看 | 亚洲国产精品一区二区尤物区 | 成人影片免费 | 欧美aaaxxxx做受视频 | 亚洲aⅴ一区二区三区 | 精品在线亚洲视频 | 国产精品专区一 | 国产免费观看视频 | 久久r精品 | 国内成人av | 久久69精品久久久久久久电影好 | 中文字幕传媒 | 欧美成人理伦片 | 国产欧美精品一区aⅴ影院 99视频国产精品免费观看 | 亚洲理论视频 | 久久观看免费视频 | 久久99精品视频 | 久久中文字幕视频 | 亚洲精品视频网址 | 丁香婷婷综合五月 | 国产在线观看免费观看 | 午夜精品久久久久久久99水蜜桃 | 揉bbb玩bbb少妇bbb | 国产91精品一区二区绿帽 | 中国一级片在线观看 | 97视频在线观看免费 | 天天色天天上天天操 | 国产色综合天天综合网 | 欧美成亚洲 | 99精品久久精品一区二区 | 超薄丝袜一二三区 | 国产做爰视频 | 欧美色图一区 | 夜夜视频欧洲 | 国产系列精品av | 免费欧美精品 | 五月天综合色 | 丁香五婷| 深夜激情影院 | 91亚洲夫妻| 黄色1级大片 | 国产一区二区午夜 | 91传媒在线看 | 国内成人精品视频 | 久久国产免费视频 | 黄色大片日本 | 久久成年人| 99久久99视频| 密桃av在线| 色wwwww | 免费一级特黄录像 | 日本三级国产 | 在线观影网站 | 免费在线观看av网站 | 黄在线免费看 | 久久免费视频7 | 亚洲欧美日韩精品久久奇米一区 | 国产精品一区久久久久 | 国产在线更新 | 中文字幕在线资源 | 亚洲日本va中文字幕 | 日韩免费视频在线观看 | 91色偷偷 | 国产综合精品久久 | 亚洲一二区视频 | av片在线观看 | 激情综合一区 | 亚洲国产99 | 日韩欧美一区二区三区黑寡妇 | 色综合久久综合网 | 久久久午夜视频 | 国产片网站 | 精品国产综合区久久久久久 | 国产第一福利网 | 免费日韩电影 | 日韩三级视频在线观看 | 色婷婷天天干 | 天天干天天操天天做 | 国产午夜精品av一区二区 | 久久伊人八月婷婷综合激情 | 天天综合久久综合 | 国产精品成人在线观看 | 国产99亚洲 | 国产精品一区二区在线免费观看 | 亚洲精品在线观看不卡 | 91精品一区二区三区久久久久久 | 狠狠干.com| 看v片| 麻豆 free xxxx movies hd | 91视频国产免费 | 久久精品视频网址 | 最新色站 | 不卡的av电影在线观看 | 精品在线亚洲视频 | 97理论电影 | 超碰人人草人人 | 一区二区三区福利 | 一区二区三区在线不卡 | 一级欧美日韩 | www.天天操| 97超碰在线播放 | 久久综合九色综合97_ 久久久 | 色播五月激情综合网 | 国产成人精品一区二区三区在线 | av网址最新 | 中文字幕视频一区二区 | 国产高清中文字幕 | 免费精品视频在线观看 | 免费成人在线网站 | 草在线 | 久久免费成人精品视频 | 日韩xxx视频| 国产h在线播放 | 中文一区在线观看 | 在线免费高清一区二区三区 | 久久国产精品精品国产色婷婷 | 日韩av在线免费播放 | 在线观看aa | 国内99视频| 精品国产一区二区三区久久 | 国语黄色片 | 精品成人国产 | 黄色av电影网 | 日韩成人在线一区二区 | 丰满少妇在线观看资源站 | 黄色在线看网站 | 久久婷婷国产 | www.夜夜骑.com| 久久精品资源 | 国产欧美综合在线观看 | 黄色av一区二区 | 国产日韩精品久久 | 午夜视频福利 | 婷婷免费在线视频 | 国产精品亚洲片夜色在线 | 免费www视频| 一级黄色网址 | 婷色| 中文字幕在线免费看 | 日日操日日干 | 国产在线永久 | 免费看黄网站在线 | 日韩免费在线一区 | 婷婷亚洲五月色综合 | 久久免费在线视频 | 免费高清在线观看成人 | 久久爱综合 | 久久久免费精品视频 | 亚洲在线视频网站 | 久久视频免费在线 | 九九在线视频免费观看 | 国产一级91 | 97激情影院| 亚洲精品国产精品国自 | 免费看片亚洲 | 中文av不卡| 日韩av成人免费看 | 色综合久久久久综合99 | 天天干人人 | 亚洲精品在线免费观看视频 | 欧美日韩免费视频 | 少妇性aaaaaaaaa视频 | 免费中文字幕视频 | 国产亚洲高清视频 | 夜夜躁狠狠躁日日躁视频黑人 | 一区二区不卡高清 | 亚洲视频一 | 免费观看国产精品视频 | 天天曰天天射 | 免费网站v| 国产精品av免费在线观看 | 视频在线一区二区三区 | 国产一区二区三区免费在线观看 | 久久精品99久久久久久 | 中文字幕资源网在线观看 | 91精品国产三级a在线观看 | 久久久久日本精品一区二区三区 | 日韩精品免费一区二区三区 | 色在线亚洲| 99精品国产一区二区三区不卡 | av网站免费看 | 国产精品成人免费精品自在线观看 | 久久综合九色欧美综合狠狠 | 伊人狠狠色丁香婷婷综合 | 日韩精品你懂的 | 麻豆免费视频观看 | 欧美精品久久久久久久久久久 | 国产精品亚州 | 综合色狠狠 | 欧美在线视频二区 | 三级动态视频在线观看 | 久久久久五月天 | 日韩av在线高清 | 国产91电影在线观看 | 欧美久久久久久久久久 | 在线综合 亚洲 欧美在线视频 | 人人狠狠综合久久亚洲 | 久久久久欠精品国产毛片国产毛生 | 日韩成人黄色av | 久久视频在线观看 | 亚洲精品一区二区三区在线观看 | 欧美日韩高清在线观看 |