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

歡迎訪問 生活随笔!

生活随笔

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

编程问答

HashMap 源码详细分析(JDK1.8)

發布時間:2025/3/21 编程问答 45 豆豆
生活随笔 收集整理的這篇文章主要介紹了 HashMap 源码详细分析(JDK1.8) 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

1. 概述

本篇文章我們來聊聊大家日常開發中常用的一個集合類 -?HashMap。HashMap 最早出現在 JDK 1.2中,底層基于散列算法實現。HashMap 允許 null 鍵和 null 值,在計算哈鍵的哈希值時,null 鍵哈希值為 0。HashMap 并不保證鍵值對的順序,這意味著在進行某些操作后,鍵值對的順序可能會發生變化。另外,需要注意的是,HashMap 是非線程安全類,在多線程環境下可能會存在問題。

在本篇文章中,我將會對 HashMap 中常用方法、重要屬性及相關方法進行分析。需要說明的是,HashMap 源碼中可分析的點很多,本文很難一一覆蓋,請見諒。

?2. 原理

上一節說到 HashMap 底層是基于散列算法實現,散列算法分為散列再探測和拉鏈式。HashMap 則使用了拉鏈式的散列算法,并在 JDK 1.8 中引入了紅黑樹優化過長的鏈表。數據結構示意圖如下:

對于拉鏈式的散列算法,其數據結構是由數組和鏈表(或樹形結構)組成。在進行增刪查等操作時,首先要定位到元素的所在桶的位置,之后再從鏈表中定位該元素。比如我們要查詢上圖結構中是否包含元素35,步驟如下:

  • 定位元素35所處桶的位置,index = 35 % 16 = 3
  • 在3號桶所指向的鏈表中繼續查找,發現35在鏈表中。
  • 上面就是 HashMap 底層數據結構的原理,HashMap 基本操作就是對拉鏈式散列算法基本操作的一層包裝。不同的地方在于 JDK 1.8 中引入了紅黑樹,底層數據結構由數組+鏈表變為了數組+鏈表+紅黑樹,不過本質并未變。好了,原理部分先講到這,接下來說說源碼實現。

    ?3. 源碼分析

    本篇文章所分析的源碼版本為 JDK 1.8。與 JDK 1.7 相比,JDK 1.8 對 HashMap 進行了一些優化。比如引入紅黑樹解決過長鏈表效率低的問題。重寫 resize 方法,移除了 alternative hashing 相關方法,避免重新計算鍵的 hash 等。不過本篇文章并不打算對這些優化進行分析,本文僅會分析 HashMap 常用的方法及一些重要屬性和相關方法。如果大家對紅黑樹感興趣,可以閱讀我的另一篇文章 -?紅黑樹詳細分析。

    ?3.1 構造方法

    ?3.1.1 構造方法分析

    HashMap 的構造方法不多,只有四個。HashMap 構造方法做的事情比較簡單,一般都是初始化一些重要變量,比如 loadFactor 和 threshold。而底層的數據結構則是延遲到插入鍵值對時再進行初始化。HashMap 相關構造方法如下:

    1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 /** 構造方法 1 */ public HashMap() {this.loadFactor = DEFAULT_LOAD_FACTOR; // all other fields defaulted }/** 構造方法 2 */ public HashMap(int initialCapacity) {this(initialCapacity, DEFAULT_LOAD_FACTOR); }/** 構造方法 3 */ public HashMap(int initialCapacity, float loadFactor) {if (initialCapacity < 0)throw new IllegalArgumentException("Illegal initial capacity: " +initialCapacity);if (initialCapacity > MAXIMUM_CAPACITY)initialCapacity = MAXIMUM_CAPACITY;if (loadFactor <= 0 || Float.isNaN(loadFactor))throw new IllegalArgumentException("Illegal load factor: " +loadFactor);this.loadFactor = loadFactor;this.threshold = tableSizeFor(initialCapacity); }/** 構造方法 4 */ public HashMap(Map<? extends K, ? extends V> m) {this.loadFactor = DEFAULT_LOAD_FACTOR;putMapEntries(m, false); }

    上面4個構造方法中,大家平時用的最多的應該是第一個了。第一個構造方法很簡單,僅將 loadFactor 變量設為默認值。構造方法2調用了構造方法3,而構造方法3仍然只是設置了一些變量。構造方法4則是將另一個 Map 中的映射拷貝一份到自己的存儲結構中來,這個方法不是很常用。

    上面就是對構造方法簡單的介紹,構造方法本身并沒什么太多東西,所以就不說了。接下來說說構造方法所初始化的幾個的變量。

    ?3.1.2 初始容量、負載因子、閾值

    我們在一般情況下,都會使用無參構造方法創建 HashMap。但當我們對時間和空間復雜度有要求的時候,使用默認值有時可能達不到我們的要求,這個時候我們就需要手動調參。在 HashMap 構造方法中,可供我們調整的參數有兩個,一個是初始容量 initialCapacity,另一個負載因子 loadFactor。通過這兩個設定這兩個參數,可以進一步影響閾值大小。但初始閾值 threshold 僅由 initialCapacity 經過移位操作計算得出。他們的作用分別如下:

    名稱用途
    initialCapacityHashMap 初始容量
    loadFactor負載因子
    threshold當前 HashMap 所能容納鍵值對數量的最大值,超過這個值,則需擴容

    相關代碼如下:

    1 2 3 4 5 6 7 8 9 10 /** The default initial capacity - MUST be a power of two. */ static final int DEFAULT_INITIAL_CAPACITY = 1 << 4;/** The load factor used when none specified in constructor. */ static final float DEFAULT_LOAD_FACTOR = 0.75f;final float loadFactor;/** The next size value at which to resize (capacity * load factor). */ int threshold;

    如果大家去看源碼,會發現 HashMap 中沒有定義 initialCapacity 這個變量。這個也并不難理解,從參數名上可看出,這個變量表示一個初始容量,只是構造方法中用一次,沒必要定義一個變量保存。但如果大家仔細看上面 HashMap 的構造方法,會發現存儲鍵值對的數據結構并不是在構造方法里初始化的。這就有個疑問了,既然叫初始容量,但最終并沒有用與初始化數據結構,那傳這個參數還有什么用呢?這個問題我先不解釋,給大家留個懸念,后面會說明。

    默認情況下,HashMap 初始容量是16,負載因子為 0.75。這里并沒有默認閾值,原因是閾值可由容量乘上負載因子計算而來(注釋中有說明),即threshold = capacity * loadFactor。但當你仔細看構造方法3時,會發現閾值并不是由上面公式計算而來,而是通過一個方法算出來的。這是不是可以說明 threshold 變量的注釋有誤呢?還是僅這里進行了特殊處理,其他地方遵循計算公式呢?關于這個疑問,這里也先不說明,后面在分析擴容方法時,再來解釋這個問題。接下來,我們來看看初始化 threshold 的方法長什么樣的的,源碼如下:

    1 2 3 4 5 6 7 8 9 10 11 12 /*** Returns a power of two size for the given target capacity.*/ static final int tableSizeFor(int cap) {int n = cap - 1;n |= n >>> 1;n |= n >>> 2;n |= n >>> 4;n |= n >>> 8;n |= n >>> 16;return (n < 0) ? 1 : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY : n + 1; }

    上面的代碼長的有點不太好看,反正我第一次看的時候不明白它想干啥。不過后來在紙上畫畫,知道了它的用途。總結起來就一句話:找到大于或等于 cap 的最小2的冪。至于為啥要這樣,后面再解釋。我們先來看看 tableSizeFor 方法的圖解:

    上面是 tableSizeFor 方法的計算過程圖,這里cap = 536,870,913 = 2<sup>29</sup> + 1,多次計算后,算出n + 1 = 1,073,741,824 = 2<sup>30</sup>。通過圖解應該可以比較容易理解這個方法的用途,這里就不多說了。

    說完了初始閾值的計算過程,再來說說負載因子(loadFactor)。對于 HashMap 來說,負載因子是一個很重要的參數,該參數反應了 HashMap 桶數組的使用情況(假設鍵值對節點均勻分布在桶數組中)。通過調節負載因子,可使 HashMap 時間和空間復雜度上有不同的表現。當我們調低負載因子時,HashMap 所能容納的鍵值對數量變少。擴容時,重新將鍵值對存儲新的桶數組里,鍵的鍵之間產生的碰撞會下降,鏈表長度變短。此時,HashMap 的增刪改查等操作的效率將會變高,這里是典型的拿空間換時間。相反,如果增加負載因子(負載因子可以大于1),HashMap 所能容納的鍵值對數量變多,空間利用率高,但碰撞率也高。這意味著鏈表長度變長,效率也隨之降低,這種情況是拿時間換空間。至于負載因子怎么調節,這個看使用場景了。一般情況下,我們用默認值就可以了。

    ?3.2 查找

    HashMap 的查找操作比較簡單,查找步驟與原理篇介紹一致,即先定位鍵值對所在的桶的位置,然后再對鏈表或紅黑樹進行查找。通過這兩步即可完成查找,該操作相關代碼如下:

    1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 public V get(Object key) {Node<K,V> e;return (e = getNode(hash(key), key)) == null ? null : e.value; }final Node<K,V> getNode(int hash, Object key) {Node<K,V>[] tab; Node<K,V> first, e; int n; K k;// 1. 定位鍵值對所在桶的位置if ((tab = table) != null && (n = tab.length) > 0 &&(first = tab[(n - 1) & hash]) != null) {if (first.hash == hash && // always check first node((k = first.key) == key || (key != null && key.equals(k))))return first;if ((e = first.next) != null) {// 2. 如果 first 是 TreeNode 類型,則調用黑紅樹查找方法if (first instanceof TreeNode)return ((TreeNode<K,V>)first).getTreeNode(hash, key);// 2. 對鏈表進行查找do {if (e.hash == hash &&((k = e.key) == key || (key != null && key.equals(k))))return e;} while ((e = e.next) != null);}}return null; }

    查找的核心邏輯是封裝在 getNode 方法中的,getNode 方法源碼我已經寫了一些注釋,應該不難看懂。我們先來看看查找過程的第一步 - 確定桶位置,其實現代碼如下:

    1 2 // index = (n - 1) & hash first = tab[(n - 1) & hash]

    這里通過(n - 1)& hash即可算出桶的在桶數組中的位置,可能有的朋友不太明白這里為什么這么做,這里簡單解釋一下。HashMap 中桶數組的大小 length 總是2的冪,此時,(n - 1) & hash?等價于對 length 取余。但取余的計算效率沒有位運算高,所以(n - 1) & hash也是一個小的優化。舉個例子說明一下吧,假設 hash = 185,n = 16。計算過程示意圖如下:

    上面的計算并不復雜,這里就不多說了。

    在上面源碼中,除了查找相關邏輯,還有一個計算 hash 的方法。這個方法源碼如下:

    1 2 3 4 5 6 7 /*** 計算鍵的 hash 值*/ static final int hash(Object key) {int h;return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16); }

    看這個方法的邏輯好像是通過位運算重新計算 hash,那么這里為什么要這樣做呢?為什么不直接用鍵的 hashCode 方法產生的 hash 呢?大家先可以思考一下,我把答案寫在下面。

    這樣做有兩個好處,我來簡單解釋一下。我們再看一下上面求余的計算圖,圖中的 hash 是由鍵的 hashCode 產生。計算余數時,由于 n 比較小,hash 只有低4位參與了計算,高位的計算可以認為是無效的。這樣導致了計算結果只與低位信息有關,高位數據沒發揮作用。為了處理這個缺陷,我們可以上圖中的 hash 高4位數據與低4位數據進行異或運算,即?hash ^ (hash >>> 4)。通過這種方式,讓高位數據與低位數據進行異或,以此加大低位信息的隨機性,變相的讓高位數據參與到計算中。此時的計算過程如下:

    在 Java 中,hashCode 方法產生的 hash 是 int 類型,32 位寬。前16位為高位,后16位為低位,所以要右移16位。

    上面所說的是重新計算 hash 的一個好處,除此之外,重新計算 hash 的另一個好處是可以增加 hash 的復雜度。當我們覆寫 hashCode 方法時,可能會寫出分布性不佳的 hashCode 方法,進而導致 hash 的沖突率比較高。通過移位和異或運算,可以讓 hash 變得更復雜,進而影響 hash 的分布性。這也就是為什么 HashMap 不直接使用鍵對象原始 hash 的原因了。

    ?3.3 遍歷

    和查找查找一樣,遍歷操作也是大家使用頻率比較高的一個操作。對于 遍歷 HashMap,我們一般都會用下面的方式:

    1 2 3 for(Object key : map.keySet()) {// do something }

    1 2 3 for(HashMap.Entry entry : map.entrySet()) {// do something }

    從上面代碼片段中可以看出,大家一般都是對 HashMap 的 key 集合或 Entry 集合進行遍歷。上面代碼片段中用 foreach 遍歷 keySet 方法產生的集合,在編譯時會轉換成用迭代器遍歷,等價于:

    1 2 3 4 5 6 Set keys = map.keySet(); Iterator ite = keys.iterator(); while (ite.hasNext()) {Object key = ite.next();// do something }

    大家在遍歷 HashMap 的過程中會發現,多次對 HashMap 進行遍歷時,遍歷結果順序都是一致的。但這個順序和插入的順序一般都是不一致的。產生上述行為的原因是怎樣的呢?大家想一下原因。我先把遍歷相關的代碼貼出來,如下:

    1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 public Set<K> keySet() {Set<K> ks = keySet;if (ks == null) {ks = new KeySet();keySet = ks;}return ks; }/*** 鍵集合*/ final class KeySet extends AbstractSet<K> {public final int size() { return size; }public final void clear() { HashMap.this.clear(); }public final Iterator<K> iterator() { return new KeyIterator(); }public final boolean contains(Object o) { return containsKey(o); }public final boolean remove(Object key) {return removeNode(hash(key), key, null, false, true) != null;}// 省略部分代碼 }/*** 鍵迭代器*/ final class KeyIterator extends HashIterator implements Iterator<K> {public final K next() { return nextNode().key; } }abstract class HashIterator {Node<K,V> next; // next entry to returnNode<K,V> current; // current entryint expectedModCount; // for fast-failint index; // current slotHashIterator() {expectedModCount = modCount;Node<K,V>[] t = table;current = next = null;index = 0;if (t != null && size > 0) { // advance to first entry // 尋找第一個包含鏈表節點引用的桶do {} while (index < t.length && (next = t[index++]) == null);}}public final boolean hasNext() {return next != null;}final Node<K,V> nextNode() {Node<K,V>[] t;Node<K,V> e = next;if (modCount != expectedModCount)throw new ConcurrentModificationException();if (e == null)throw new NoSuchElementException();if ((next = (current = e).next) == null && (t = table) != null) {// 尋找下一個包含鏈表節點引用的桶do {} while (index < t.length && (next = t[index++]) == null);}return e;}//省略部分代碼 }

    如上面的源碼,遍歷所有的鍵時,首先要獲取鍵集合KeySet對象,然后再通過 KeySet 的迭代器KeyIterator進行遍歷。KeyIterator 類繼承自HashIterator類,核心邏輯也封裝在 HashIterator 類中。HashIterator 的邏輯并不復雜,在初始化時,HashIterator 先從桶數組中找到包含鏈表節點引用的桶。然后對這個桶指向的鏈表進行遍歷。遍歷完成后,再繼續尋找下一個包含鏈表節點引用的桶,找到繼續遍歷。找不到,則結束遍歷。舉個例子,假設我們遍歷下圖的結構:

    HashIterator 在初始化時,會先遍歷桶數組,找到包含鏈表節點引用的桶,對應圖中就是3號桶。隨后由 nextNode 方法遍歷該桶所指向的鏈表。遍歷完3號桶后,nextNode 方法繼續尋找下一個不為空的桶,對應圖中的7號桶。之后流程和上面類似,直至遍歷完最后一個桶。以上就是 HashIterator 的核心邏輯的流程,對應下圖:

    遍歷上圖的最終結果是?19 -> 3 -> 35 -> 7 -> 11 -> 43 -> 59,為了驗證正確性,簡單寫點測試代碼跑一下看看。測試代碼如下:

    1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 /*** 應在 JDK 1.8 下測試,其他環境下不保證結果和上面一致*/ public class HashMapTest {@Testpublic void testTraversal() {HashMap<Integer, String> map = new HashMap(16);map.put(7, "");map.put(11, "");map.put(43, "");map.put(59, "");map.put(19, "");map.put(3, "");map.put(35, "");System.out.println("遍歷結果:");for (Integer key : map.keySet()) {System.out.print(key + " -> ");}} }

    遍歷結果如下:

    在本小節的最后,拋兩個問題給大家。在 JDK 1.8 版本中,為了避免過長的鏈表對 HashMap 性能的影響,特地引入了紅黑樹優化性能。但在上面的源碼中并沒有發現紅黑樹遍歷的相關邏輯,這是為什么呢?對于被轉換成紅黑樹的鏈表該如何遍歷呢?大家可以先想想,然后可以去源碼或本文后續章節中找答案。

    ?3.4 插入

    ?3.4.1 插入邏輯分析

    通過前兩節的分析,大家對 HashMap 低層的數據結構應該了然于心了。即使我不說,大家也應該能知道 HashMap 的插入流程是什么樣的了。首先肯定是先定位要插入的鍵值對屬于哪個桶,定位到桶后,再判斷桶是否為空。如果為空,則將鍵值對存入即可。如果不為空,則需將鍵值對接在鏈表最后一個位置,或者更新鍵值對。這就是 HashMap 的插入流程,是不是覺得很簡單。當然,大家先別高興。這只是一個簡化版的插入流程,真正的插入流程要復雜不少。首先 HashMap 是變長集合,所以需要考慮擴容的問題。其次,在 JDK 1.8 中,HashMap 引入了紅黑樹優化過長鏈表,這里還要考慮多長的鏈表需要進行優化,優化過程又是怎樣的問題。引入這里兩個問題后,大家會發現原本簡單的操作,現在略顯復雜了。在本節中,我將先分析插入操作的源碼,擴容、樹化(鏈表轉為紅黑樹,下同)以及其他和樹結構相關的操作,隨后將在獨立的兩小結中進行分析。接下來,先來看一下插入操作的源碼:

    1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 public V put(K key, V value) {return putVal(hash(key), key, value, false, true); }final V putVal(int hash, K key, V value, boolean onlyIfAbsent,boolean evict) {Node<K,V>[] tab; Node<K,V> p; int n, i;// 初始化桶數組 table,table 被延遲到插入新數據時再進行初始化if ((tab = table) == null || (n = tab.length) == 0)n = (tab = resize()).length;// 如果桶中不包含鍵值對節點引用,則將新鍵值對節點的引用存入桶中即可if ((p = tab[i = (n - 1) & hash]) == null)tab[i] = newNode(hash, key, value, null);else {Node<K,V> e; K k;// 如果鍵的值以及節點 hash 等于鏈表中的第一個鍵值對節點時,則將 e 指向該鍵值對if (p.hash == hash &&((k = p.key) == key || (key != null && key.equals(k))))e = p;// 如果桶中的引用類型為 TreeNode,則調用紅黑樹的插入方法else if (p instanceof TreeNode) e = ((TreeNode<K,V>)p).putTreeVal(this, tab, hash, key, value);else {// 對鏈表進行遍歷,并統計鏈表長度for (int binCount = 0; ; ++binCount) {// 鏈表中不包含要插入的鍵值對節點時,則將該節點接在鏈表的最后if ((e = p.next) == null) {p.next = newNode(hash, key, value, null);// 如果鏈表長度大于或等于樹化閾值,則進行樹化操作if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1sttreeifyBin(tab, hash);break;}// 條件為 true,表示當前鏈表包含要插入的鍵值對,終止遍歷if (e.hash == hash &&((k = e.key) == key || (key != null && key.equals(k))))break;p = e;}}// 判斷要插入的鍵值對是否存在 HashMap 中if (e != null) { // existing mapping for keyV oldValue = e.value;// onlyIfAbsent 表示是否僅在 oldValue 為 null 的情況下更新鍵值對的值if (!onlyIfAbsent || oldValue == null)e.value = value;afterNodeAccess(e);return oldValue;}}++modCount;// 鍵值對數量超過閾值時,則進行擴容if (++size > threshold)resize();afterNodeInsertion(evict);return null; }

    插入操作的入口方法是?put(K,V),但核心邏輯在V putVal(int, K, V, boolean, boolean)?方法中。putVal 方法主要做了這么幾件事情:

  • 當桶數組 table 為空時,通過擴容的方式初始化 table
  • 查找要插入的鍵值對是否已經存在,存在的話根據條件判斷是否用新值替換舊值
  • 如果不存在,則將鍵值對鏈入鏈表中,并根據鏈表長度決定是否將鏈表轉為紅黑樹
  • 判斷鍵值對數量是否大于閾值,大于的話則進行擴容操作
  • 以上就是 HashMap 插入的邏輯,并不是很復雜,這里就不多說了。接下來來分析一下擴容機制。

    ?3.4.2 擴容機制

    在 Java 中,數組的長度是固定的,這意味著數組只能存儲固定量的數據。但在開發的過程中,很多時候我們無法知道該建多大的數組合適。建小了不夠用,建大了用不完,造成浪費。如果我們能實現一種變長的數組,并按需分配空間就好了。好在,我們不用自己實現變長數組,Java 集合框架已經實現了變長的數據結構。比如 ArrayList 和 HashMap。對于這類基于數組的變長數據結構,擴容是一個非常重要的操作。下面就來聊聊 HashMap 的擴容機制。

    在詳細分析之前,先來說一下擴容相關的背景知識:

    在 HashMap 中,桶數組的長度均是2的冪,閾值大小為桶數組長度與負載因子的乘積。當 HashMap 中的鍵值對數量超過閾值時,進行擴容。

    HashMap 的擴容機制與其他變長集合的套路不太一樣,HashMap 按當前桶數組長度的2倍進行擴容,閾值也變為原來的2倍(如果計算過程中,閾值溢出歸零,則按閾值公式重新計算)。擴容之后,要重新計算鍵值對的位置,并把它們移動到合適的位置上去。以上就是 HashMap 的擴容大致過程,接下來我們來看看具體的實現:

    1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 final Node<K,V>[] resize() {Node<K,V>[] oldTab = table;int oldCap = (oldTab == null) ? 0 : oldTab.length;int oldThr = threshold;int newCap, newThr = 0;// 如果 table 不為空,表明已經初始化過了if (oldCap > 0) {// 當 table 容量超過容量最大值,則不再擴容if (oldCap >= MAXIMUM_CAPACITY) {threshold = Integer.MAX_VALUE;return oldTab;} // 按舊容量和閾值的2倍計算新容量和閾值的大小else if ((newCap = oldCap << 1) < MAXIMUM_CAPACITY &&oldCap >= DEFAULT_INITIAL_CAPACITY)newThr = oldThr << 1; // double threshold} else if (oldThr > 0) // initial capacity was placed in threshold/** 初始化時,將 threshold 的值賦值給 newCap,* HashMap 使用 threshold 變量暫時保存 initialCapacity 參數的值*/ newCap = oldThr;else { // zero initial threshold signifies using defaults/** 調用無參構造方法時,桶數組容量為默認容量,* 閾值為默認容量與默認負載因子乘積*/newCap = DEFAULT_INITIAL_CAPACITY;newThr = (int)(DEFAULT_LOAD_FACTOR * DEFAULT_INITIAL_CAPACITY);}// newThr 為 0 時,按閾值計算公式進行計算if (newThr == 0) {float ft = (float)newCap * loadFactor;newThr = (newCap < MAXIMUM_CAPACITY && ft < (float)MAXIMUM_CAPACITY ?(int)ft : Integer.MAX_VALUE);}threshold = newThr;// 創建新的桶數組,桶數組的初始化也是在這里完成的Node<K,V>[] newTab = (Node<K,V>[])new Node[newCap];table = newTab;if (oldTab != null) {// 如果舊的桶數組不為空,則遍歷桶數組,并將鍵值對映射到新的桶數組中for (int j = 0; j < oldCap; ++j) {Node<K,V> e;if ((e = oldTab[j]) != null) {oldTab[j] = null;if (e.next == null)newTab[e.hash & (newCap - 1)] = e;else if (e instanceof TreeNode)// 重新映射時,需要對紅黑樹進行拆分((TreeNode<K,V>)e).split(this, newTab, j, oldCap);else { // preserve orderNode<K,V> loHead = null, loTail = null;Node<K,V> hiHead = null, hiTail = null;Node<K,V> next;// 遍歷鏈表,并將鏈表節點按原順序進行分組do {next = e.next;if ((e.hash & oldCap) == 0) {if (loTail == null)loHead = e;elseloTail.next = e;loTail = e;}else {if (hiTail == null)hiHead = e;elsehiTail.next = e;hiTail = e;}} while ((e = next) != null);// 將分組后的鏈表映射到新桶中if (loTail != null) {loTail.next = null;newTab[j] = loHead;}if (hiTail != null) {hiTail.next = null;newTab[j + oldCap] = hiHead;}}}}}return newTab; }

    上面的源碼有點長,希望大家耐心看懂它的邏輯。上面的源碼總共做了3件事,分別是:

  • 計算新桶數組的容量 newCap 和新閾值 newThr
  • 根據計算出的 newCap 創建新的桶數組,桶數組 table 也是在這里進行初始化的
  • 將鍵值對節點重新映射到新的桶數組里。如果節點是 TreeNode 類型,則需要拆分紅黑樹。如果是普通節點,則節點按原順序進行分組。
  • 上面列的三點中,創建新的桶數組就一行代碼,不用說了。接下來,來說說第一點和第三點,先說說 newCap 和 newThr 計算過程。該計算過程對應 resize 源碼的第一和第二個條件分支,如下:

    1 2 3 4 5 6 7 8 9 10 11 12 // 第一個條件分支 if ( oldCap > 0) {// 嵌套條件分支if (oldCap >= MAXIMUM_CAPACITY) {...}else if ((newCap = oldCap << 1) < MAXIMUM_CAPACITY &&oldCap >= DEFAULT_INITIAL_CAPACITY) {...} } else if (oldThr > 0) {...} else {...}// 第二個條件分支 if (newThr == 0) {...}

    通過這兩個條件分支對不同情況進行判斷,進而算出不同的容量值和閾值。它們所覆蓋的情況如下:

    分支一:

    條件覆蓋情況備注
    oldCap > 0桶數組 table 已經被初始化?
    oldThr > 0threshold > 0,且桶數組未被初始化調用?HashMap(int) 和 HashMap(int, float) 構造方法時會產生這種情況,此種情況下 newCap = oldThr,newThr 在第二個條件分支中算出
    oldCap == 0 &&?oldThr == 0桶數組未被初始化,且?threshold 為 0調用?HashMap() 構造方法會產生這種情況。

    這里把oldThr > 0情況單獨拿出來說一下。在這種情況下,會將 oldThr 賦值給 newCap,等價于newCap = threshold = tableSizeFor(initialCapacity)。我們在初始化時傳入的 initialCapacity 參數經過 threshold 中轉最終賦值給了 newCap。這也就解答了前面提的一個疑問:initialCapacity 參數沒有被保存下來,那么它怎么參與桶數組的初始化過程的呢?

    嵌套分支:

    條件覆蓋情況備注
    oldCap >= 230桶數組容量大于或等于最大桶容量 230這種情況下不再擴容
    newCap < 230?&& oldCap > 16新桶數組容量小于最大值,且舊桶數組容量大于 16該種情況下新閾值 newThr = oldThr << 1,移位可能會導致溢出

    這里簡單說明一下移位導致的溢出情況,當 loadFactor小數位為 0,整數位可被2整除且大于等于8時,在某次計算中就可能會導致 newThr 溢出歸零。見下圖:

    分支二:

    條件覆蓋情況備注
    newThr == 0第一個條件分支未計算 newThr 或嵌套分支在計算過程中導致 newThr 溢出歸零?

    說完 newCap 和 newThr 的計算過程,接下來再來分析一下鍵值對節點重新映射的過程。

    在 JDK 1.8 中,重新映射節點需要考慮節點類型。對于樹形節點,需先拆分紅黑樹再映射。對于鏈表類型節點,則需先對鏈表進行分組,然后再映射。需要的注意的是,分組后,組內節點相對位置保持不變。關于紅黑樹拆分的邏輯將會放在下一小節說明,先來看看鏈表是怎樣進行分組映射的。

    我們都知道往底層數據結構中插入節點時,一般都是先通過模運算計算桶位置,接著把節點放入桶中即可。事實上,我們可以把重新映射看做插入操作。在 JDK 1.7 中,也確實是這樣做的。但在 JDK 1.8 中,則對這個過程進行了一定的優化,邏輯上要稍微復雜一些。在詳細分析前,我們先來回顧一下 hash 求余的過程:

    上圖中,桶數組大小 n = 16,hash1 與 hash2 不相等。但因為只有后4位參與求余,所以結果相等。當桶數組擴容后,n 由16變成了32,對上面的 hash 值重新進行映射:

    擴容后,參與模運算的位數由4位變為了5位。由于兩個 hash 第5位的值是不一樣,所以兩個 hash 算出的結果也不一樣。上面的計算過程并不難理解,繼續往下分析。

    假設我們上圖的桶數組進行擴容,擴容后容量 n = 16,重新映射過程如下:

    依次遍歷鏈表,并計算節點?hash & oldCap?的值。如下圖所示

    如果值為0,將 loHead 和 loTail 指向這個節點。如果后面還有節點 hash & oldCap 為0的話,則將節點鏈入 loHead 指向的鏈表中,并將 loTail 指向該節點。如果值為非0的話,則讓 hiHead 和 hiTail 指向該節點。完成遍歷后,可能會得到兩條鏈表,此時就完成了鏈表分組:

    最后再將這兩條鏈接存放到相應的桶中,完成擴容。如下圖:

    從上圖可以發現,重新映射后,兩條鏈表中的節點順序并未發生變化,還是保持了擴容前的順序。以上就是 JDK 1.8 中 HashMap 擴容的代碼講解。另外再補充一下,JDK 1.8 版本下 HashMap 擴容效率要高于之前版本。如果大家看過 JDK 1.7 的源碼會發現,JDK 1.7 為了防止因 hash 碰撞引發的拒絕服務攻擊,在計算 hash 過程中引入隨機種子。以增強 hash 的隨機性,使得鍵值對均勻分布在桶數組中。在擴容過程中,相關方法會根據容量判斷是否需要生成新的隨機種子,并重新計算所有節點的 hash。而在 JDK 1.8 中,則通過引入紅黑樹替代了該種方式。從而避免了多次計算 hash 的操作,提高了擴容效率。

    本小節的內容講就先講到這,接下來,來講講鏈表與紅黑樹相互轉換的過程。

    ?3.4.3 鏈表樹化、紅黑樹鏈化與拆分

    JDK 1.8 對 HashMap 實現進行了改進。最大的改進莫過于在引入了紅黑樹處理頻繁的碰撞,代碼復雜度也隨之上升。比如,以前只需實現一套針對鏈表操作的方法即可。而引入紅黑樹后,需要另外實現紅黑樹相關的操作。紅黑樹是一種自平衡的二叉查找樹,本身就比較復雜。本篇文章中并不打算對紅黑樹展開介紹,本文僅會介紹鏈表樹化需要注意的地方。至于紅黑樹詳細的介紹,如果大家有興趣,可以參考我的另一篇文章 -?紅黑樹詳細分析。

    在展開說明之前,先把樹化的相關代碼貼出來,如下:

    1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 static final int TREEIFY_THRESHOLD = 8;/*** 當桶數組容量小于該值時,優先進行擴容,而不是樹化*/ static final int MIN_TREEIFY_CAPACITY = 64;static final class TreeNode<K,V> extends LinkedHashMap.Entry<K,V> {TreeNode<K,V> parent; // red-black tree linksTreeNode<K,V> left;TreeNode<K,V> right;TreeNode<K,V> prev; // needed to unlink next upon deletionboolean red;TreeNode(int hash, K key, V val, Node<K,V> next) {super(hash, key, val, next);} }/*** 將普通節點鏈表轉換成樹形節點鏈表*/ final void treeifyBin(Node<K,V>[] tab, int hash) {int n, index; Node<K,V> e;// 桶數組容量小于 MIN_TREEIFY_CAPACITY,優先進行擴容而不是樹化if (tab == null || (n = tab.length) < MIN_TREEIFY_CAPACITY)resize();else if ((e = tab[index = (n - 1) & hash]) != null) {// hd 為頭節點(head),tl 為尾節點(tail)TreeNode<K,V> hd = null, tl = null;do {// 將普通節點替換成樹形節點TreeNode<K,V> p = replacementTreeNode(e, null);if (tl == null)hd = p;else {p.prev = tl;tl.next = p;}tl = p;} while ((e = e.next) != null); // 將普通鏈表轉成由樹形節點鏈表if ((tab[index] = hd) != null)// 將樹形鏈表轉換成紅黑樹hd.treeify(tab);} }TreeNode<K,V> replacementTreeNode(Node<K,V> p, Node<K,V> next) {return new TreeNode<>(p.hash, p.key, p.value, next); }

    在擴容過程中,樹化要滿足兩個條件:

  • 鏈表長度大于等于 TREEIFY_THRESHOLD
  • 桶數組容量大于等于 MIN_TREEIFY_CAPACITY
  • 第一個條件比較好理解,這里就不說了。這里來說說加入第二個條件的原因,個人覺得原因如下:

    當桶數組容量比較小時,鍵值對節點 hash 的碰撞率可能會比較高,進而導致鏈表長度較長。這個時候應該優先擴容,而不是立馬樹化。畢竟高碰撞率是因為桶數組容量較小引起的,這個是主因。容量小時,優先擴容可以避免一些列的不必要的樹化過程。同時,桶容量較小時,擴容會比較頻繁,擴容時需要拆分紅黑樹并重新映射。所以在桶容量比較小的情況下,將長鏈表轉成紅黑樹是一件吃力不討好的事。

    回到上面的源碼中,我們繼續看一下 treeifyBin 方法。該方法主要的作用是將普通鏈表轉成為由 TreeNode 型節點組成的鏈表,并在最后調用 treeify 是將該鏈表轉為紅黑樹。TreeNode 繼承自 Node 類,所以 TreeNode 仍然包含 next 引用,原鏈表的節點順序最終通過 next 引用被保存下來。我們假設樹化前,鏈表結構如下:

    HashMap 在設計之初,并沒有考慮到以后會引入紅黑樹進行優化。所以并沒有像 TreeMap 那樣,要求鍵類實現 comparable 接口或提供相應的比較器。但由于樹化過程需要比較兩個鍵對象的大小,在鍵類沒有實現 comparable 接口的情況下,怎么比較鍵與鍵之間的大小了就成了一個棘手的問題。為了解決這個問題,HashMap 是做了三步處理,確保可以比較出兩個鍵的大小,如下:

  • 比較鍵與鍵之間 hash 的大小,如果 hash 相同,繼續往下比較
  • 檢測鍵類是否實現了 Comparable 接口,如果實現調用 compareTo 方法進行比較
  • 如果仍未比較出大小,就需要進行仲裁了,仲裁方法為 tieBreakOrder(大家自己看源碼吧)
  • tie break 是網球術語,可以理解為加時賽的意思,起這個名字還是挺有意思的。

    通過上面三次比較,最終就可以比較出孰大孰小。比較出大小后就可以構造紅黑樹了,最終構造出的紅黑樹如下:

    橙色的箭頭表示 TreeNode 的 next 引用。由于空間有限,prev 引用未畫出。可以看出,鏈表轉成紅黑樹后,原鏈表的順序仍然會被引用仍被保留了(紅黑樹的根節點會被移動到鏈表的第一位),我們仍然可以按遍歷鏈表的方式去遍歷上面的紅黑樹。這樣的結構為后面紅黑樹的切分以及紅黑樹轉成鏈表做好了鋪墊,我們繼續往下分析。

    ?紅黑樹拆分

    擴容后,普通節點需要重新映射,紅黑樹節點也不例外。按照一般的思路,我們可以先把紅黑樹轉成鏈表,之后再重新映射鏈表即可。這種處理方式是大家比較容易想到的,但這樣做會損失一定的效率。不同于上面的處理方式,HashMap 實現的思路則是上好佳(上好佳請把廣告費打給我)。如上節所說,在將普通鏈表轉成紅黑樹時,HashMap 通過兩個額外的引用 next 和 prev 保留了原鏈表的節點順序。這樣再對紅黑樹進行重新映射時,完全可以按照映射鏈表的方式進行。這樣就避免了將紅黑樹轉成鏈表后再進行映射,無形中提高了效率。

    以上就是紅黑樹拆分的邏輯,下面看一下具體實現吧:

    1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 // 紅黑樹轉鏈表閾值 static final int UNTREEIFY_THRESHOLD = 6;final void split(HashMap<K,V> map, Node<K,V>[] tab, int index, int bit) {TreeNode<K,V> b = this;// Relink into lo and hi lists, preserving orderTreeNode<K,V> loHead = null, loTail = null;TreeNode<K,V> hiHead = null, hiTail = null;int lc = 0, hc = 0;/* * 紅黑樹節點仍然保留了 next 引用,故仍可以按鏈表方式遍歷紅黑樹。* 下面的循環是對紅黑樹節點進行分組,與上面類似*/for (TreeNode<K,V> e = b, next; e != null; e = next) {next = (TreeNode<K,V>)e.next;e.next = null;if ((e.hash & bit) == 0) {if ((e.prev = loTail) == null)loHead = e;elseloTail.next = e;loTail = e;++lc;}else {if ((e.prev = hiTail) == null)hiHead = e;elsehiTail.next = e;hiTail = e;++hc;}}if (loHead != null) {// 如果 loHead 不為空,且鏈表長度小于等于 6,則將紅黑樹轉成鏈表if (lc <= UNTREEIFY_THRESHOLD)tab[index] = loHead.untreeify(map);else {tab[index] = loHead;/* * hiHead == null 時,表明擴容后,* 所有節點仍在原位置,樹結構不變,無需重新樹化*/if (hiHead != null) loHead.treeify(tab);}}// 與上面類似if (hiHead != null) {if (hc <= UNTREEIFY_THRESHOLD)tab[index + bit] = hiHead.untreeify(map);else {tab[index + bit] = hiHead;if (loHead != null)hiHead.treeify(tab);}} }

    從源碼上可以看得出,重新映射紅黑樹的邏輯和重新映射鏈表的邏輯基本一致。不同的地方在于,重新映射后,會將紅黑樹拆分成兩條由 TreeNode 組成的鏈表。如果鏈表長度小于 UNTREEIFY_THRESHOLD,則將鏈表轉換成普通鏈表。否則根據條件重新將 TreeNode 鏈表樹化。舉個例子說明一下,假設擴容后,重新映射上圖的紅黑樹,映射結果如下:

    ?紅黑樹鏈化

    前面說過,紅黑樹中仍然保留了原鏈表節點順序。有了這個前提,再將紅黑樹轉成鏈表就簡單多了,僅需將 TreeNode 鏈表轉成 Node 類型的鏈表即可。相關代碼如下:

    1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 final Node<K,V> untreeify(HashMap<K,V> map) {Node<K,V> hd = null, tl = null;// 遍歷 TreeNode 鏈表,并用 Node 替換for (Node<K,V> q = this; q != null; q = q.next) {// 替換節點類型Node<K,V> p = map.replacementNode(q, null);if (tl == null)hd = p;elsetl.next = p;tl = p;}return hd; }Node<K,V> replacementNode(Node<K,V> p, Node<K,V> next) {return new Node<>(p.hash, p.key, p.value, next); }

    上面的代碼并不復雜,不難理解,這里就不多說了。到此擴容相關內容就說完了,不知道大家理解沒。

    ?3.5 刪除

    如果大家堅持看完了前面的內容,到本節就可以輕松一下。當然,前提是不去看紅黑樹的刪除操作。不過紅黑樹并非本文講解重點,本節中也不會介紹紅黑樹相關內容,所以大家不用擔心。

    HashMap 的刪除操作并不復雜,僅需三個步驟即可完成。第一步是定位桶位置,第二步遍歷鏈表并找到鍵值相等的節點,第三步刪除節點。相關源碼如下:

    1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 public V remove(Object key) {Node<K,V> e;return (e = removeNode(hash(key), key, null, false, true)) == null ?null : e.value; }final Node<K,V> removeNode(int hash, Object key, Object value,boolean matchValue, boolean movable) {Node<K,V>[] tab; Node<K,V> p; int n, index;if ((tab = table) != null && (n = tab.length) > 0 &&// 1. 定位桶位置(p = tab[index = (n - 1) & hash]) != null) {Node<K,V> node = null, e; K k; V v;// 如果鍵的值與鏈表第一個節點相等,則將 node 指向該節點if (p.hash == hash &&((k = p.key) == key || (key != null && key.equals(k))))node = p;else if ((e = p.next) != null) { // 如果是 TreeNode 類型,調用紅黑樹的查找邏輯定位待刪除節點if (p instanceof TreeNode)node = ((TreeNode<K,V>)p).getTreeNode(hash, key);else {// 2. 遍歷鏈表,找到待刪除節點do {if (e.hash == hash &&((k = e.key) == key ||(key != null && key.equals(k)))) {node = e;break;}p = e;} while ((e = e.next) != null);}}// 3. 刪除節點,并修復鏈表或紅黑樹if (node != null && (!matchValue || (v = node.value) == value ||(value != null && value.equals(v)))) {if (node instanceof TreeNode)((TreeNode<K,V>)node).removeTreeNode(this, tab, movable);else if (node == p)tab[index] = node.next;elsep.next = node.next;++modCount;--size;afterNodeRemoval(node);return node;}}return null; }

    刪除操作本身并不復雜,有了前面的基礎,理解起來也就不難了,這里就不多說了。

    ?3.6 其他細節

    前面的內容分析了 HashMap 的常用操作及相關的源碼,本節內容再補充一點其他方面的東西。

    ?被 transient 所修飾 table 變量

    如果大家細心閱讀 HashMap 的源碼,會發現桶數組 table 被申明為 transient。transient 表示易變的意思,在 Java 中,被該關鍵字修飾的變量不會被默認的序列化機制序列化。我們再回到源碼中,考慮一個問題:桶數組 table 是 HashMap 底層重要的數據結構,不序列化的話,別人還怎么還原呢?

    這里簡單說明一下吧,HashMap 并沒有使用默認的序列化機制,而是通過實現readObject/writeObject兩個方法自定義了序列化的內容。這樣做是有原因的,試問一句,HashMap 中存儲的內容是什么?不用說,大家也知道是鍵值對。所以只要我們把鍵值對序列化了,我們就可以根據鍵值對數據重建 HashMap。有的朋友可能會想,序列化 table 不是可以一步到位,后面直接還原不就行了嗎?這樣一想,倒也是合理。但序列化 talbe 存在著兩個問題:

  • table 多數情況下是無法被存滿的,序列化未使用的部分,浪費空間
  • 同一個鍵值對在不同 JVM 下,所處的桶位置可能是不同的,在不同的 JVM 下反序列化 table 可能會發生錯誤。
  • 以上兩個問題中,第一個問題比較好理解,第二個問題解釋一下。HashMap 的get/put/remove等方法第一步就是根據 hash 找到鍵所在的桶位置,但如果鍵沒有覆寫 hashCode 方法,計算 hash 時最終調用 Object 中的 hashCode 方法。但 Object 中的 hashCode 方法是 native 型的,不同的 JVM 下,可能會有不同的實現,產生的 hash 可能也是不一樣的。也就是說同一個鍵在不同平臺下可能會產生不同的 hash,此時再對在同一個 table 繼續操作,就會出現問題。

    綜上所述,大家應該能明白 HashMap 不序列化 table 的原因了。

    ?3.7 總結

    本章對 HashMap 常見操作相關代碼進行了詳細分析,并在最后補充了一些其他細節。在本章中,插入操作一節的內容說的最多,主要是因為插入操作涉及的點特別多,一環扣一環。包含但不限于“table 初始化、擴容、樹化”等,總體來說,插入操作分析起來難度還是很大的。好在,最后分析完了。

    本章篇幅雖比較大,但仍未把 HashMap 所有的點都分析到。比如,紅黑樹的增刪查等操作。當然,我個人看來,以上的分析已經夠了。畢竟大家是類庫的使用者而不是設計者,沒必要去弄懂每個細節。所以如果某些細節實在看不懂的話就跳過吧,對我們開發來說,知道 HashMap 大致原理即可。

    好了,本章到此結束。

    ?4、寫在最后

    寫到這里終于可以松一口氣了,這篇文章前前后后花了我一周多的時間。在我寫這篇文章之前,對 HashMap 認識僅限于原理層面,并未深入了解。一開始,我覺得關于 HashMap 沒什么好寫的,畢竟大家對 HashMap 多少都有一定的了解。但等我深入閱讀 HashMap 源碼后,發現之前的認知是錯的。不是沒什么可寫的,而是可寫的點太多了,不知道怎么寫了。JDK 1.8 版本的 HashMap 實現上比之前版本要復雜的多,想弄懂眾多的細節難度還是不小的。僅自己弄懂還不夠,還要寫出來,難度就更大了,本篇文章基本上是在邊讀源碼邊寫的狀態下完成的。由于時間和能力有限,加之文章篇幅比較大,很難保證不出錯分析過程及配圖不出錯。如果有錯誤,希望大家指出來,我會及時修改,這里先謝謝大家。

    好了,本文就到這里了,謝謝大家的閱讀!

    ?參考

    • JDK 源碼中 HashMap 的 hash 方法原理是什么?- 知乎
    • Java 8系列之重新認識HashMap - 美團技術博客
    • python內置的hash函數對于字符串來說,每次得到的值不一樣?- 知乎
    • Java中HashMap關鍵字transient的疑惑 - segmentFault)
    • 本文鏈接:?https://www.tianxiaobo.com/2018/01/18/HashMap-源碼詳細分析-JDK1-8/

    from:http://www.tianxiaobo.com/2018/01/18/HashMap-%E6%BA%90%E7%A0%81%E8%AF%A6%E7%BB%86%E5%88%86%E6%9E%90-JDK1-8/?

    總結

    以上是生活随笔為你收集整理的HashMap 源码详细分析(JDK1.8)的全部內容,希望文章能夠幫你解決所遇到的問題。

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

    天无日天天操天天干 | 婷婷色综 | 久草免费在线观看 | 草久视频在线观看 | 五月激情久久久 | 91精品国产电影 | 成人黄色在线播放 | 92精品国产成人观看免费 | 亚洲三级黄 | 亚洲精品男人的天堂 | 天天干天天色2020 | 欧美激情精品久久久久久变态 | 国产高清视频在线播放 | 精品久久综合 | 视频国产在线 | 亚洲v精品 | 国产精品久久久久久久久久久久午夜 | 国产精品久久久久久久午夜片 | 国产一区欧美在线 | 日韩免费观看一区二区三区 | a级成人毛片 | 午夜精品福利一区二区 | 国产一级二级在线观看 | 97成人免费| 日韩中文久久 | 国产精品亚洲综合久久 | 久久久成人精品 | 久久久久久久99 | 免费成人在线观看视频 | 丁香六月婷| 欧洲视频一区 | 国产视频一区二区三区在线 | 国产小视频在线播放 | 视频一区二区在线观看 | av韩国在线 | 日韩专区在线播放 | 99久久久国产精品 | 人人涩 | 国产精品一区在线观看你懂的 | 97精产国品一二三产区在线 | 日韩在线视频网址 | 久久久精品网站 | 玖玖视频网 | 国产精品久久99综合免费观看尤物 | 国产成人精品av | 一级淫片在线观看 | www.狠狠操.com | 91视频三区 | www.色综合.com| 国产成人精品免费在线观看 | 欧美韩国在线 | 国产1级毛片 | 欧美日韩一区二区三区在线观看视频 | 久久久久久久久毛片 | 国产亚洲综合性久久久影院 | 日本女人b | 九九视频免费在线观看 | 人人操日日干 | 久久久久一区二区三区四区 | 狠狠干夜夜爽 | 中文字幕资源网在线观看 | 91在线你懂的 | 一区二区三区电影在线播 | 在线播放视频一区 | 免费在线中文字幕 | 手机在线观看国产精品 | 在线日韩中文字幕 | 欧美精品在线免费 | 亚洲日韩中文字幕在线播放 | 色婷婷狠狠操 | 日韩精品观看 | 成年人黄色大片在线 | 久久伦理影院 | www.狠狠色| 中文字幕在线观看第二页 | 免费看的av片 | 国产视频在线观看一区 | 黄色一级网| 九九久久久| 久久综合狠狠综合久久狠狠色综合 | 中文字幕日韩在线播放 | 日本性xxx| 久久精品波多野结衣 | 福利久久| 天天爽夜夜爽精品视频婷婷 | 国产黄在线播放 | 亚洲成人中文在线 | 亚洲japanese制服美女 | 二区三区av| 欧美一级视频免费 | 国产在线观看高清视频 | 国产福利久久 | 91免费观看视频网站 | 色多多视频在线观看 | 国产视频黄 | 亚洲成av人影院 | 免费看黄网站在线 | 激情欧美在线观看 | 天天干天天草天天爽 | 在线观看国产亚洲 | 久久伦理视频 | 97色免费视频 | 在线看成人片 | 在线导航av | 欧洲性视频 | 免费观看成人 | 国产色视频123区 | 狠狠色噜噜狠狠狠狠 | 久久精美视频 | a在线一区 | 黄色免费网战 | 日韩区在线观看 | 亚洲精品视频第一页 | 欧美日韩不卡在线 | 精品福利网| 久久久久欧美精品 | 欧美做受高潮电影o | 99久久精品国产一区二区成人 | 国产中文字幕在线播放 | 中文字幕欧美日韩va免费视频 | 亚洲欧洲日韩 | 久久男人视频 | 欧美久久成人 | 97在线精品视频 | 日本99干网 | 97在线影院 | 美女视频久久 | 99re亚洲国产精品 | 久久精品9| 91精品1区2区 | 91污视频在线观看 | 国产乱老熟视频网88av | 国产日韩在线观看一区 | 国产精品中文字幕av | 夜夜夜夜猛噜噜噜噜噜初音未来 | 欧美少妇的秘密 | 91tv国产成人福利 | 国产精品一级在线 | 国产啊v在线观看 | 亚洲一二三久久 | 日韩欧美一区二区三区黑寡妇 | 99激情网| 这里只有精品视频在线 | 欧美日韩精品免费观看 | 九七在线视频 | 日韩欧美在线观看 | 五月综合色婷婷 | 91传媒激情理伦片 | 中文字幕专区高清在线观看 | 在线免费观看国产精品 | av电影在线免费观看 | 人人澡人人爽欧一区 | 欧美一级久久 | 最新av免费在线观看 | 国产一级免费观看 | 粉嫩av一区二区三区四区 | 久久精品久久99精品久久 | 日日插日日干 | 日韩 国产 | 久久精彩视频 | 日本久久成人 | 国产男女免费完整视频 | 亚洲欧美激情精品一区二区 | 男女拍拍免费视频 | 米奇狠狠狠888 | 在线视频观看成人 | 精品主播网红福利资源观看 | 中文字幕在线网址 | 精品国产一区二区三区久久久 | 中国一级片免费看 | 精品久久久久一区二区国产 | 91免费高清视频 | 亚洲 成人 欧美 | 99麻豆久久久国产精品免费 | 亚洲精品电影在线 | 国产黄色片在线 | 中文字幕在线影院 | 国产精品久久电影观看 | 奇米影视8888在线观看大全免费 | 深夜免费小视频 | 手机成人在线 | 欧洲色吧 | 成年人免费在线观看 | 超碰日韩 | 精精国产xxxx视频在线播放 | 人人狠狠综合久久亚洲 | 亚洲午夜久久久久久久久久久 | 久久99最新地址 | 欧美一级免费高清 | 在线观看精品视频 | 玖草影院 | 色爱区综合激月婷婷 | 91香蕉视频720p| 天堂久久电影网 | 日韩精品久久一区二区 | 亚洲欧美一区二区三区孕妇写真 | 日韩欧美xx | 成人av网页| 亚洲成人精品影院 | 国产小视频在线免费观看视频 | 一区二区男女 | 狠狠干婷婷色 | 在线观看国产麻豆 | 亚洲欧美视频在线观看 | 欧美在线free | 91 在线视频 | 成人一区二区在线 | 美女福利视频一区二区 | 日p视频在线观看 | 91喷水 | 国产精品高潮在线观看 | 国产精品久久久久久久久软件 | 伊人天天综合 | 久久伊人精品一区二区三区 | 日日日日干 | 九九热久久免费视频 | 国产视频欧美视频 | 国产成人久久精品77777综合 | 亚洲精品视频在线播放 | 久久精品国产一区二区三 | 亚洲成人动漫在线观看 | 免费情趣视频 | 狠狠综合网 | 色丁香综合| 黄色.com | 99免费在线观看视频 | av看片在线观看 | 欧美日韩在线观看一区二区三区 | 91天堂影院 | 久久国产精品一区二区三区 | 亚洲专区 国产精品 | 中文在线a天堂 | 人人爽人人爽人人爽人人爽 | 久操伊人 | 久久久国产精品麻豆 | 国产精品久久免费看 | 激情深爱.com | 7777xxxx| 涩涩网站在线 | 三日本三级少妇三级99 | 日韩免费看的电影 | 日本在线观看中文字幕 | 精品国产美女 | 天天干天天怕 | 国产精品亚洲a | 东方av在 | 国产精品大片 | 欧美日韩精品免费观看视频 | 91桃色在线播放 | 国产精品久久电影观看 | 一区二区三区高清在线 | av一级久久| 亚洲国产精品日韩 | 日日天天干 | 免费在线观看视频a | 韩国视频一区二区三区 | 免费视频久久 | 国内成人精品2018免费看 | 午夜成人免费影院 | 免费看一级特黄a大片 | 精品国产欧美一区二区三区不卡 | 亚洲欧美日韩国产一区二区三区 | 天天综合网 天天 | 久草亚洲视频 | 五月婷影院 | 国产亚洲精品久久久久久电影 | 日韩一级成人av | 国产精品久久久久高潮 | 国产精品久久视频 | 国产一级一片免费播放放 | 欧美日本中文字幕 | 国内少妇自拍视频一区 | 日韩高清不卡一区二区三区 | 国产精品免费视频久久久 | 中文在线字幕免费观看 | 国产精久久久久久妇女av | 黄网站色视频免费观看 | 久草网在线观看 | 亚洲影院天堂 | 久久综合狠狠综合久久综合88 | 精品999久久久 | 在线va视频 | 久久久久久影视 | 日韩资源在线播放 | 可以免费看av | 国产精品区一区 | 蜜臀av性久久久久蜜臀av | 欧美激情视频一二三区 | 天天操夜夜爱 | 日本性高潮视频 | 狠狠操夜夜 | 国产亚洲成av人片在线观看桃 | 国产视频 久久久 | 欧美精品在线一区二区 | 久久一区二区免费视频 | 成年人黄色免费网站 | 91免费看片黄 | 日韩av专区| 碰碰影院| 日韩在线网址 | 国产美女被啪进深处喷白浆视频 | 亚洲国产一区在线观看 | 黄网站app在线观看免费视频 | 国产亚洲精品成人av久久ww | 久久久91精品国产一区二区精品 | 久久久久久国产精品免费 | 精品久久国产 | 成人资源在线观看 | 成人在线黄色电影 | 亚洲精品乱码久久久久久久久久 | 日韩电影久久久 | 97免费在线观看视频 | 91麻豆精品国产 | 国产不卡视频在线播放 | 国产精品久免费的黄网站 | 中文字幕精品三区 | 激情综合电影网 | 91爱爱网址| 国产福利91精品一区二区三区 | 色婷婷六月 | 亚洲综合少妇 | 国产精品男女啪啪 | 国产淫片免费看 | 久久伊人精品一区二区三区 | 亚洲欧美怡红院 | 国产精品mv | 黄色av大片| 一区二区精品久久 | 狠狠操夜夜 | 久久精品一区 | 蜜臀aⅴ国产精品久久久国产 | 一区中文字幕电影 | 免费99视频| 成人中文字幕av | 欧美精品二区 | 婷婷福利影院 | 亚洲乱码在线观看 | 久久99精品久久久久久秒播蜜臀 | 国产精品扒开做爽爽的视频 | 精品女同一区二区三区在线观看 | 三级视频国产 | 欧美日韩免费观看一区=区三区 | 天天干天天做天天爱 | 国产三级精品三级在线观看 | 狠狠久久婷婷 | 国产69精品久久99的直播节目 | 亚洲免费精彩视频 | 精品在线二区 | 亚洲激精日韩激精欧美精品 | 亚洲开心色 | av中文字幕免费在线观看 | 97视频人人免费看 | 免费看v片网站 | 中文字幕电影高清在线观看 | 欧美日本不卡视频 | 国产999精品久久久久久 | 国产免费久久精品 | 国产九九在线 | 少妇高潮流白浆在线观看 | a视频免费在线观看 | 欧美精品免费在线 | 亚洲精品99久久久久久 | 黄色片视频在线观看 | 九九欧美 | www色婷婷com| 久久精品日产第一区二区三区乱码 | 在线观看视频黄 | 伊人久久电影网 | ww视频在线观看 | 友田真希av | 麻豆综合网| 夜夜骑日日| 欧美一级日韩三级 | av福利在线播放 | 国产亚洲精品久久久久久久久久久久 | 五月婷婷激情综合网 | 国产中文字幕第一页 | 久久久网页 | 久久精选视频 | 五月天com| 天天五月天色 | 免费情趣视频 | 亚洲电影在线看 | 精品一区二区免费 | 狠狠干美女 | 欧美另类一二三四区 | 一区 二区 精品 | 天天摸日日摸人人看 | 99国产精品免费网站 | 久久人人爽爽人人爽人人片av | 日韩精品在线播放 | 国内精品福利视频 | 色婷婷激情 | 国产在线91在线电影 | 特级西西444www大胆高清无视频 | 成人一级片免费看 | 亚洲一区日韩精品 | 911国产在线观看 | 国产剧情一区二区 | 91日韩在线 | 色成人亚洲网 | 激情xxxx| 超碰97免费 | 香蕉视频国产在线观看 | 肉色欧美久久久久久久免费看 | 日本精品久久久久中文字幕5 | 日韩免费不卡视频 | 亚洲国产精品va在线看 | 亚洲欧美成aⅴ人在线观看 四虎在线观看 | 夜夜躁狠狠燥 | 91成人观看 | 狠狠躁日日躁夜夜躁av | 特级免费毛片 | 亚洲一区精品二人人爽久久 | 国产一级视频在线免费观看 | 日韩欧美网址 | 日韩黄色中文字幕 | 日韩欧在线 | 欧美大片在线看免费观看 | 免费高清在线一区 | 国产精品美女久久久久久免费 | 亚洲午夜精品久久久久久久久 | 国产一区二区在线免费观看 | 国内成人av | 五月婷婷电影网 | 日韩免费网站 | 91在线91| 久人人| 又长又大又黑又粗欧美 | 97超级碰碰| 人人澡人人爱 | 三级av在线免费观看 | 日韩av手机在线看 | 日韩精品视频久久 | bayu135国产精品视频 | 国产va饥渴难耐女保洁员在线观看 | 麻豆国产视频下载 | 日韩成人欧美 | 91视频91自拍 | 91精品一区国产高清在线gif | 看毛片网站| 波多野结衣在线观看视频 | wwxxx日本| 久草久草在线观看 | 91超级碰| 日本性生活免费看 | 欧美精品久久久久久久免费 | 99视| 在线观看免费av网站 | av黄色在线播放 | 黄色av三级在线 | 亚洲精品国产片 | 99久久99视频只有精品 | 亚洲天堂精品视频 | 黄色的视频网站 | 欧美日韩视频在线 | 国产日产精品久久久久快鸭 | 日本精品一区二区三区在线播放视频 | 99成人在线视频 | 六月色丁香 | 成人免费视频视频在线观看 免费 | 色五月激情五月 | 99视频这里只有 | 激情五月综合网 | 久久久久久久久爱 | 久久视频在线观看中文字幕 | 在线不卡视频 | 国产伦精品一区二区三区高清 | 97夜夜澡人人爽人人免费 | 国产人成看黄久久久久久久久 | 中文字幕av在线电影 | 毛片888 | 色七七亚洲影院 | 中文字幕 国产视频 | 亚洲精品免费看 | 伊人狠狠干| 国产99久久99热这里精品5 | 色婷婷色 | 精品一区91 | 日韩欧美一区二区三区在线观看 | 色中射| 十八岁免进欧美 | 国产高清视频在线播放一区 | 97国产在线观看 | 黄色小说免费观看 | 四虎成人免费观看 | 久草国产在线观看 | 久久久久免费网站 | 精品美女在线视频 | 亚洲综合视频在线观看 | 亚洲精品av中文字幕在线在线 | 五月婷婷激情 | 欧美性色黄大片在线观看 | 日韩精品一区二区三区视频播放 | 中日韩欧美精彩视频 | 色视频一区 | 偷拍区另类综合在线 | 久草在线 | 日韩成年视频 | 97视频在线播放 | 97人人模人人爽人人喊网 | 亚洲成人黄色网址 | 久久免费视频99 | 久久国产网站 | 久久精品欧美 | 国产视频二区三区 | 亚洲欧洲日韩 | 正在播放国产91 | 国产色拍拍拍拍在线精品 | 国产一区二区播放 | 亚洲在线黄色 | 国产免费又粗又猛又爽 | 欧美日韩国产精品久久 | 精品国产一区二区三区av性色 | 99视频网址 | 亚洲免费婷婷 | 精品久久视频 | 欧美一级免费片 | 久久99中文字幕 | 91精品久久久久久久久久久久久 | 国产精品青草综合久久久久99 | 在线观看日韩av | 免费观看国产成人 | 特级免费毛片 | 日韩视频www| 中文字幕av免费 | 91精品国产成人 | 青青河边草免费直播 | 精品视频久久久久久 | 久久久999免费视频 日韩网站在线 | 美女视频黄是免费的 | 亚州国产视频 | 二区三区av | 麻豆va一区二区三区久久浪 | 欧美一区二区三区免费观看 | 日本一区二区三区免费观看 | 在线99热 | 中文字幕视频一区二区 | 精品国产视频在线观看 | 在线看片中文字幕 | 在线视频免费观看 | 91视频在线网址 | 天天干天天射天天爽 | 国产精品美女毛片真酒店 | 国产中文字幕视频 | 日韩精品免费专区 | 五月婷婷综合在线视频 | 精品黄色在线 | 久久久久免费精品国产小说色大师 | 久久人人爽人人爽人人片 | 久久久久久久影视 | 婷婷视频在线观看 | 999久久久免费视频 午夜国产在线观看 | 中文字幕免费观看视频 | 日本一区二区高清不卡 | 久久久久久99精品 | 99热手机在线| 一级黄色片网站 | 久久玖| av黄免费看 | 中文字幕欧美日韩va免费视频 | 亚洲精品乱码久久久久久按摩 | 亚洲午夜电影网 | 国产精品资源 | 婷婷丁香五| 国产最顶级的黄色片在线免费观看 | 成人一区二区三区在线 | 黄色在线看网站 | 视频三区| 国产一区二区免费看 | 黄色一级在线视频 | 超碰人人av | 国产精品久久久久久久免费 | 日韩 精品 一区 国产 麻豆 | 亚洲一区二区黄色 | 日韩黄色大片在线观看 | 精品久久久久久综合 | 欧美a级一区二区 | 欧美在线观看视频一区二区 | 丁香一区二区 | 午夜私人影院久久久久 | 亚洲国产操 | 夜色资源站wwwcom | www.在线观看视频 | 玖玖爱在线观看 | 精品一区二区免费在线观看 | 午夜久久久久久久久久影院 | 国产亚洲情侣一区二区无 | 国产香蕉视频在线播放 | 激情视频综合网 | 国产成人在线一区 | 在线免费观看视频一区二区三区 | 国产免费专区 | 96香蕉视频 | 人人干人人草 | 成人黄色在线播放 | 日韩欧美高清免费 | 色五丁香 | 成人av影视在线 | 国产91免费观看 | 国产一区视频免费在线观看 | 精品一区二区三区香蕉蜜桃 | 日韩欧美在线免费观看 | 日韩精品视频在线观看网址 | 久久免费视频网 | 97在线视频免费播放 | 毛片网站在线看 | 欧美片网站yy | 久久久久久影视 | 久久理论视频 | 国产精品视屏 | 五月天天在线 | 黄色国产大片 | 91精品专区| 欧美日韩不卡一区二区 | 亚洲精品伦理在线 | 国产短视频在线播放 | 国产精品美乳一区二区免费 | 精品一区电影国产 | 亚洲视频免费视频 | 欧美日韩免费观看一区=区三区 | 91精品在线免费观看 | 久久久久日本精品一区二区三区 | 96超碰在线 | 99久久精品国产毛片 | 香蕉久久国产 | 国产精品综合在线 | 国产在线 一区二区三区 | 日韩在线高清免费视频 | 亚洲国产99| 91成人免费| 特黄色大片| 激情电影在线观看 | 97超碰人人网 | 特片网久久 | 91精品国产乱码久久桃 | 91热视频在线观看 | 欧美伦理一区 | 国产一区视频在线播放 | 中文字幕高清免费日韩视频在线 | 九七视频在线观看 | 亚洲精品国偷自产在线99热 | 国产69精品久久久久9999apgf | 日本视频久久久 | 超碰人人在 | 91资源在线视频 | 天天看天天干 | 天天射天天干 | 久久不卡视频 | 91中文字幕永久在线 | 久久久久国产精品免费 | 国产伦理一区二区三区 | 久久午夜鲁丝片 | 亚洲国产人午在线一二区 | 日韩字幕在线观看 | 欧美日韩国产精品一区二区三区 | 插久久 | 国产福利免费在线观看 | 国产精品国产自产拍高清av | 国产又黄又爽又猛视频日本 | 久久综合中文字幕 | 久久久久视 | 国产成人精品亚洲 | 欧美日韩免费观看一区二区三区 | 免费的黄色的网站 | 久久久精品 一区二区三区 国产99视频在线观看 | 九九综合久久 | 500部大龄熟乱视频使用方法 | 精品福利网站 | 成人毛片一区二区三区 | 国产精品一区二区久久久久 | 在线精品视频免费播放 | 久保带人 | 懂色av懂色av粉嫩av分享吧 | 亚洲黄色免费在线看 | 亚洲干视频在线观看 | 精品久久久久久久久久久久久久久久久久 | 亚洲精品国偷拍自产在线观看蜜桃 | 国语精品久久 | 久久精品人人做人人综合老师 | 日本黄色免费电影网站 | 日韩乱码中文字幕 | 免费视频资源 | 青青河边草免费观看 | 精品视频99 | 久久国产网站 | 欧美色图88| 国产精品初高中精品久久 | av中文天堂 | 国产在线资源 | 久久国产高清 | 激情电影影院 | 9在线观看免费高清完整版 玖玖爱免费视频 | 五月导航| 天海翼一区二区三区免费 | 黄色av影院| 黄色小视频在线观看免费 | 九九涩涩av台湾日本热热 | 国产一二三区av | 久草在线高清视频 | 91大神精品视频在线观看 | 日韩av片免费在线观看 | 久精品视频在线观看 | 日韩在线 | 久久久久成人免费 | 最新日韩中文字幕 | 91精品国自产在线 | 毛片网站观看 | 91亚洲国产 | 免费看国产精品 | 99r在线播放 | 欧美日韩高清在线一区 | 国产成人精品一区二区三区福利 | 亚洲精品高清一区二区三区四区 | 国产男男gay做爰 | 最新av在线网站 | 夜夜夜夜猛噜噜噜噜噜初音未来 | 中文字幕最新精品 | 久久精品这里热有精品 | 亚洲日本韩国一区二区 | www亚洲一区 | 精品久久久久久亚洲综合网 | 中文字幕一区二区三区精华液 | 久久国产精品精品国产色婷婷 | 一区二区三区在线免费观看视频 | 久久久99精品免费观看乱色 | 国产资源网 | 国产精品99久久久久 | 日本精品一区二区三区在线播放视频 | 911久久香蕉国产线看观看 | 国产亚洲精品日韩在线tv黄 | 蜜臀久久99精品久久久无需会员 | 中文字幕在 | 亚洲精品成人在线 | 天天视频亚洲 | 五月婷婷色丁香 | 蜜臀aⅴ国产精品久久久国产 | 日本爱爱片 | 草在线| 久久久午夜视频 | se视频网址| 欧美日韩精品影院 | 最新免费av在线 | 久久99精品久久只有精品 | 国产直播av | 日韩电影中文,亚洲精品乱码 | 国产精品麻| 91av成人| 久久国产精品久久w女人spa | 99久久久久久久久久 | 午夜三级影院 | 中文字幕一区二区三区四区 | 色诱亚洲精品久久久久久 | 在线91av| 91精品国产一区二区三区 | 成人影音av | 欧美黑人巨大xxxxx | 国产高潮久久 | 午夜精品久久久久久久99 | 三级免费黄色 | 精品在线观看国产 | 四虎免费在线观看视频 | 天天摸日日摸人人看 | 极品国产91在线网站 | 国产午夜精品免费一区二区三区视频 | 久久伊人操 | 国产色在线观看 | 成人午夜片av在线看 | 91久久偷偷做嫩草影院 | 一区二区三区日韩在线观看 | 亚洲色图美腿丝袜 | 99视频免费观看 | 在线看的av网站 | 亚洲一区久久久 | 精品国产aⅴ一区二区三区 在线直播av | 日韩色一区二区三区 | 五月婷婷婷婷婷 | 久久爱992xxoo | 国产精品一区二区吃奶在线观看 | 婷婷色五 | 九热在线 | 国产一级二级av | 久久久久久国产一区二区三区 | 9久久精品 | 国产精品毛片一区 | 男女全黄一级一级高潮免费看 | av丝袜在线 | 国产黄免费 | 久久成人国产精品免费软件 | 97人人添人澡人人爽超碰动图 | 色吧久久 | 国产精品完整版 | 免费视频成人 | 成年人视频在线观看免费 | 美女免费av | 成人a在线| 狠狠的干狠狠的操 | 国产又粗又猛又色又黄视频 | 色婷婷视频 | 国产美女视频网站 | 最新真实国产在线视频 | 黄污网站在线观看 | 国产视频手机在线 | 九九热99视频 | 丁香一区二区 | 91视频在线免费 | 99久久久久久久久久 | 久久久久免费精品国产 | 亚洲国产字幕 | 久青草影院| 久久久www成人免费毛片麻豆 | av不卡免费在线观看 | 国产亚洲欧美精品久久久久久 | 精品1区二区 | 欧美亚洲免费在线一区 | 色狠狠干 | jizzjizzjizz亚洲 | 色在线中文字幕 | 午夜婷婷网 | 国产区欧美 | 免费一级毛毛片 | 在线免费精品视频 | 成人黄色电影在线 | 国产亚洲在线视频 | 欧美在线观看小视频 | 久久久五月天 | 狂野欧美激情性xxxx欧美 | 成人免费网站在线观看 | 免费在线观看毛片网站 | 精品国产视频一区 | 国产成人免费高清 | 最新中文在线视频 | 你操综合 | 五月婷婷六月丁香激情 | 最近2019年日本中文免费字幕 | a在线v | 国产免费一区二区三区网站免费 | 国产韩国日本高清视频 | 亚洲成人黄色网址 | av免费观看在线 | 狠色狠色综合久久 | 五月激情久久久 | 丝袜制服综合网 | 91porny九色在线播放 | 97视频在线免费 | 五月天婷婷在线观看视频 | 91视频麻豆 | 久久人人爽人人爽人人片 | 日韩av免费在线看 | 欧美国产一区二区 | 日韩av中文字幕在线免费观看 | 国产亚洲精品美女久久 | 中文字幕中文字幕在线中文字幕三区 | 国产成人1区 | 亚洲 欧美变态 另类 综合 | 日韩在线视 | 成人一级免费电影 | 中文在线免费一区三区 | 久久久久亚洲精品中文字幕 | 国产日韩欧美在线一区 | 97超碰伊人 | 97福利| 在线免费观看黄色大片 | 亚洲国产精品资源 | 97超碰国产精品女人人人爽 | 在线影院中文字幕 | 日韩av免费在线电影 | www.777奇米| 日韩精品久久久久久 | 精品字幕在线 | 国产精品第一 | 国内精品久久久久久久久久 | 午夜视频二区 | 激情开心站 | 午夜精品久久久久久久久久久久 | 91精品国产一区二区在线观看 | 国产精品一区二区三区观看 | 午夜私人影院 | 国产精品久久电影观看 | 成人午夜电影网站 | 国产91精品在线播放 | 久久久久伦理电影 | 国产在线精品一区二区 | 99综合电影在线视频 | 亚洲电影av在线 | 成人午夜剧场在线观看 | 国产精品一区二区av影院萌芽 | 天天操天天综合网 | 99精品国产99久久久久久福利 | 五月婷婷综合网 | 天天色中文| 黄色片网站av | 亚洲精品中文在线观看 | 中文字幕色网站 | 婷婷丁香激情综合 | 精品久久久久久亚洲 | 欧亚日韩精品一区二区在线 | 91在线精品观看 | 成人av一区二区兰花在线播放 | 精品在线视频播放 | 日日日日日 | 欧美日韩国产一区二区三区 | 婷婷色伊人 | 日日操天天操狠狠操 | 97超碰国产精品女人人人爽 | 涩涩成人在线 | 久久五月婷婷丁香 | 久久久污 | 玖玖在线视频观看 | 国产精品一区免费在线观看 | 亚洲日本va午夜在线电影 | 大片网站久久 | 久久久www成人免费精品 | 久久欧美在线电影 | 亚洲视频一级 | 久久嗨| 中文在线www| 91免费网址 | 亚洲人成免费网站 | 国产资源av | 永久av免费在线观看 | 久久看视频 | 狠狠干狠狠插 | 在线视频在线观看 | 久久久久在线观看 | 日韩av片无码一区二区不卡电影 | 91av中文字幕 | 亚洲国产精品999 | 91精品小视频 | 92国产精品久久久久首页 | 手机在线视频福利 | 国内一级片在线观看 | 99久久99精品 | 91亚色视频在线观看 | 国产在线观 | 97福利视频 | 久久久久久久久网站 | 国产精品高清免费在线观看 | 91成人国产 | 国产激情免费 | 91大神免费视频 | 美女视频黄是免费的 | 99精品在线视频播放 | 国产精品自在线 | 97超级碰 | 91精品国产网站 | 在线三级av | 亚洲精品午夜视频 | 在线不卡中文字幕播放 | 国产黄色视 | 国内精品久久久久国产 | 国产老熟 | 青草草在线视频 | 91精品一区二区三区久久久久久 | 国产中文在线视频 | 免费视频久久久 | 久久亚洲精品电影 | 国产精品私人影院 | 丁香久久五月 | 免费不卡中文字幕视频 | 福利电影一区二区 | 久久精品网站视频 | 三日本三级少妇三级99 | 五月宗合网 | 五月天com| www.色午夜| 国内精品久久久久 | 亚洲精品国产日韩 | 久久视频在线视频 | 国产视频资源 | 国产视频 亚洲视频 | 91香蕉视频 mp4| 涩涩爱夜夜爱 | 国产精品久久久久久久久久久久久久 | 日韩一区二区三 | 天天插一插 | 激情五月在线观看 | 国外成人在线视频网站 | 日日夜夜精品视频 | 午夜精品一区二区三区视频免费看 | 亚洲我射av| 免费看网站在线 | 久久国产精品99久久久久久丝袜 | 日本h视频在线观看 | 国产黄网站在线观看 | 91欧美精品 | 91九色视频在线 | 韩日电影在线观看 | 欧美日韩久久久 | 欧洲精品久久久久毛片完整版 | 国产成人av网站 | 久人人| 久久国语| 美女黄频在线观看 | 狠狠干干| 色综合久久88色综合天天人守婷 | 在线观看av国产 | 成年人免费av网站 |