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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > c/c++ >内容正文

c/c++

c++ map 获取key列表_好未来Golang源码系列一:Map实现原理分析

發布時間:2024/7/19 c/c++ 40 豆豆
生活随笔 收集整理的這篇文章主要介紹了 c++ map 获取key列表_好未来Golang源码系列一:Map实现原理分析 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

分享老師:學而思網校 郭雨田

一、map的結構與設計原理

golang中map是一個kv對集合。底層使用hash table,用鏈表來解決沖突 ,出現沖突時,不是每一個key都申請一個結構通過鏈表串起來,而是以bmap為最小粒度掛載,一個bmap可以放8個kv。在哈希函數的選擇上,會在程序啟動時,檢測 cpu 是否支持 aes,如果支持,則使用 aes hash,否則使用 memhash每個map的底層結構是hmap,是有若干個結構為bmap的bucket組成的數組。每個bucket底層都采用鏈表結構。接下來,我們來詳細看下map的結構:

// A header for a Go map.type hmap struct { count int // 元素個數 flags uint8 B uint8 // 擴容常量相關字段B是buckets數組的長度的對數 2^B noverflow uint16 // 溢出的bucket個數 hash0 uint32 // hash seed buckets unsafe.Pointer // buckets 數組指針 oldbuckets unsafe.Pointer // 結構擴容的時候用于賦值的buckets數組 nevacuate uintptr // 搬遷進度 extra *mapextra // 用于擴容的指針}type mapextra struct { overflow *[]*bmap oldoverflow *[]*bmap nextOverflow *bmap}// A bucket for a Go map.type bmap struct { tophash [bucketCnt]uint8 // len為8的數組}//底層定義的常量 const ( // Maximum number of key/value pairs a bucket can hold. bucketCntBits = 3 bucketCnt = 1 << bucketCntBits)

但這只是表面(src/runtime/hashmap.go)的結構,編譯期間會給它加料,動態地創建一個新的結構:

type bmap struct { topbits [8]uint8 keys [8]keytype values [8]valuetype pad uintptr overflow uintptr}

hmap和bmap的結構是這樣的 :

bmap?就是我們常說的“桶”,桶里面會最多裝 8 個 key,這些 key 之所以會落入同一個桶,是因為它們經過哈希計算后,哈希結果是“一類”的,關于key的定位我們在map的查詢和賦值中詳細說明。在桶內,又會根據 key 計算出來的 hash 值的高 8 位來決定 key 到底落入桶內的哪個位置(一個桶內最多有8個位置)。當 map 的 key 和 value 都不是指針,并且 size 都小于 128 字節的情況下,會把 bmap 標記為不含指針,這樣可以避免 gc 時掃描整個 hmap。但是,我們看 bmap 其實有一個 overflow 的字段,是指針類型的,破壞了 bmap 不含指針的設想,這時會把 overflow 移動到 hmap的extra 字段來。這部分我們在分析擴容操作的時候再詳細說明。下面我們看下bmap的內部組成圖:

HOBHash?指的就是 top hash,每個bucket中topHash唯一。key 和 value 是各自放在一起的,并不是?key/value/...?這樣的形式。可以省略掉 padding 字段,節省內存空間。例如,有這樣一個類型的 map:map[int64]int8,如果按照?key/value...?這樣的模式存儲,那在每一個 key/value 對之后都要額外 padding 7 個字節;而將所有的 key,value 分別綁定到一起,這種形式?key/key/.../value/value/...,則只需要在最后添加 padding,每個 bucket 設計成最多只能放 8 個 key-value 對,如果有第 9 個 key-value 落入當前的 bucket,那就需要再構建一個 bucket ,通過?overflow?指針連接起來。

二、map操作底層原理分析

1、map初始化:

方法1:var m map[string]string // 聲明變量 --nil map 支持查詢 返回類型默認值 賦值、delete操作會panicm = make(map[string]string, 10) // 初始化 --empty map 可以進行賦值操作了方法2:m := make(map[string]string,10) // 容量參數可省略方法3:m := map[string]string{ // 通過直接賦值進行初始化 "test": "test", "name": "lili", "age": "one", }第一步:入參校驗,判斷key的類型是否合法,必須為可比較類型。

第二步:底層調用makemap函數,計算得到合適的B,map容量最多可容納6.5*2^B個元素,6.5為裝載因子閾值常量。裝載因子的計算公式是:裝載因子=填入表中的元素個數/散列表的長度裝載因子越大,說明空閑位置越少,沖突越多,散列表的性能會下降。func makemap(t *maptype, hint int, h *hmap) *hmap {//邊界校驗 if hint < 0 || hint > int(maxSliceCap(t.bucket.size)) { hint = 0 }// initialize Hmap if h == nil { h = new(hmap) }//生成hash種子 h.hash0 = fastrand() // find size parameter which will hold the requested # of elements B := uint8(0)//計算得到合適的B for overLoadFactor(hint, B) { B++ } h.B = B // allocate initial hash table // if B == 0, the buckets field is allocated lazily later (in mapassign) // If hint is large zeroing this memory could take a while.//申請桶空間 if h.B != 0 { var nextOverflow *bmap h.buckets, nextOverflow = makeBucketArray(t, h.B, nil) if nextOverflow != nil { h.extra = new(mapextra) h.extra.nextOverflow = nextOverflow } } return h}//常量loadFactorNum=13 ,loadFactorDen=2func overLoadFactor(count int, B uint8) bool { return count > bucketCnt && uintptr(count) > loadFactorNum*(bucketShift(B)/loadFactorDen)

makemap函數會通過?fastrand?創建一個隨機的哈希種子,然后根據傳入的?hint?計算出需要的最小需要的桶的數量,最后再使用?makeBucketArray創建用于保存桶的數組,這個方法其實就是根據傳入的?B?計算出的需要創建的桶數量在內存中分配一片連續的空間用于存儲數據,在創建桶的過程中還會額外創建一些用于保存溢出數據的桶,數量是?2^(B-4)?個。初始化完成返回hmap指針。

2、查找操作

Go 語言中讀取 map 有兩種語法:帶 comma 和 不帶 comma。當要查詢的 key 不在 map 里,帶 comma 的用法會返回一個 bool 型變量提示 key 是否在 map 中;而不帶 comma 的語句則會返回一個 value 類型的零值。如果 value 是 int 型就會返回 0,如果 value 是 string 類型,就會返回空字符串。

value := m["name"]fmt.Printf("value:%s", value)value, ok := m["name"] if ok { fmt.Printf("value:%s", value) }

兩種語法對應到底層兩個不同的函數,那么在底層是如何定位到key的呢?稍后我們對函數進行源碼分析。

func mapaccess1(t *maptype, h *hmap, key unsafe.Pointer) unsafe.Pointerfunc mapaccess2(t *maptype, h *hmap, key unsafe.Pointer) (unsafe.Pointer, bool)

key的定位:

key 經過哈希計算后得到哈希值,共 64 個 bit 位(64位機,32位機就不討論了,現在主流都是64位機),計算它到底要落在哪個桶時,只會用到最后 B 個 bit 位。還記得前面提到過的 B 嗎?如果 B = 5,那么桶的數量,也就是 buckets 數組的長度是 2^5 = 32。例如,現在有一個 key 經過哈希函數計算后,得到的哈希結果是:

用最后的 5 個 bit 位,也就是?01010,值為 10,也就是 10 號桶。這個操作實際上就是取余操作,但是取余開銷太大,所以代碼實現上用的位操作代替。再用哈希值的高 8 位,找到此 key 在 bucket 中的位置,這是在尋找已有的 key。最開始桶內還沒有 key,新加入的 key 會找到第一個空位放入。buckets 編號就是桶編號,當兩個不同的 key 落在同一個桶中,也就是發生了哈希沖突。沖突的解決手段是用鏈表法:在 bucket 中,從前往后找到第一個空位。這樣,在查找某個 key 時,先找到對應的桶,再去遍歷 bucket 中的 key。

上圖中,假定 B = 5,所以 bucket 總數就是 2^5 = 32。首先計算出待查找 key 的哈希,使用低 5 位?00110,找到對應的 6 號 bucket,使用高 8 位?10010111,對應十進制 151,在 6 號 bucket 中尋找 tophash 值(HOB hash)為 151 的 key,找到了 2 號槽位,這樣整個查找過程就結束了。如果在 bucket 中沒找到,并且 overflow 不為空,還要繼續去 overflow bucket 中尋找,直到找到或是所有的 key 槽位都找遍了,包括所有的 overflow bucket。接下來我們看下底層函數源碼:func mapaccess1(t *maptype, h *hmap, key unsafe.Pointer) unsafe.Pointer { //... // 如果 h 什么都沒有,返回零值 if h == nil || h.count == 0 { return unsafe.Pointer(&zeroVal[0]) } // 寫和讀沖突 if h.flags&hashWriting != 0 { throw("concurrent map read and map write") } // 不同類型 key 使用的 hash 算法在編譯期確定 alg := t.key.alg // 計算哈希值,并且加入 hash0 引入隨機性 hash := alg.hash(key, uintptr(h.hash0)) // 比如 B=5,那 m 就是31,二進制是全 1 // 求 bucket num 時,將 hash 與 m 相與, // 達到 bucket num 由 hash 的低 8 位決定的效果 m := bucketMask(h.B) // b 就是 bucket 的地址 b := (*bmap)(add(h.buckets, (hash&m)*uintptr(t.bucketsize))) // oldbuckets 不為 nil,說明發生了擴容 if c := h.oldbuckets; c != nil { // 如果不是同 size 擴容(看后面擴容的內容) // 對應條件 1 的解決方案 if !h.sameSizeGrow() { // 新 bucket 數量是老的 2 倍 m >>= 1 } // 求出 key 在老的 map 中的 bucket 位置 oldb := (*bmap)(add(c, (hash&m)*uintptr(t.bucketsize))) // 如果 oldb 沒有搬遷到新的 bucket // 那就在老的 bucket 中尋找 if !evacuated(oldb) { b = oldb } } // 計算出高 8 位的 hash // 相當于右移 56 位,只取高8位 top := tophash(hash) //開始尋找key for ; b != nil; b = b.overflow(t) { // 遍歷 8 個 bucket for i := uintptr(0); i < bucketCnt; i++ { // tophash 不匹配,繼續 if b.tophash[i] != top { continue } // tophash 匹配,定位到 key 的位置 k := add(unsafe.Pointer(b), dataOffset+i*uintptr(t.keysize)) // key 是指針 if t.indirectkey { // 解引用 k = *((*unsafe.Pointer)(k)) } // 如果 key 相等 if alg.equal(key, k) { // 定位到 value 的位置 v := add(unsafe.Pointer(b), dataOffset+bucketCnt*uintptr(t.keysize)+i*uintptr(t.valuesize)) // value 解引用 if t.indirectvalue { v = *((*unsafe.Pointer)(v)) } return v } } } return unsafe.Pointer(&zeroVal[0])}

這里我們再詳細分析下key/value值是如何獲取的:

// key 定位公式k :=add(unsafe.Pointer(b),dataOffset+i*uintptr(t.keysize))// value 定位公式v:= add(unsafe.Pointer(b),dataOffset+bucketCnt*uintptr(t.keysize)+i*uintptr(t.valuesize))//對于 bmap 起始地址的偏移:dataOffset = unsafe.Offsetof(struct{ b bmap v int64}{}.v)

bucket 里 key 的起始地址就是 unsafe.Pointer(b)+dataOffset。第 i 個 key 的地址就要在此基礎上跨過 i 個 key 的大小;而我們又知道,value 的地址是在所有 key 之后,因此第 i 個 value 的地址還需要加上所有 key 的偏移。

3、賦值操作 m := make(map[int32]int32) m[0] = 6666666

接下來我們將分成幾個部分去看看底層在賦值的時候,進行了什么操作:

第一階段:校驗和初始化

func mapassign(t *maptype, h *hmap, key unsafe.Pointer) unsafe.Pointer { //判斷 hmap 是否已經初始化(是否為 nil) if h == nil { panic(plainError("assignment to entry in nil map")) } //... //判斷是否并發讀寫 map,若是則拋出異常 if h.flags&hashWriting != 0 { throw("concurrent map writes") } //根據 key 的不同類型調用不同的 hash 方法計算得出 hash 值 alg := t.key.alg hash := alg.hash(key, uintptr(h.hash0)) //設置 flags 標志位,表示有一個 goroutine 正在寫入數據。因為 alg.hash 有可能出現 panic 導致異常 h.flags |= hashWriting //判斷 buckets 是否為 nil,若是則調用 newobject 根據當前 bucket 大小進行分配 //初始化時沒有初始 buckets,那么它在第一次賦值時就會對 buckets 分配 if h.buckets == nil { h.buckets = newobject(t.bucket) // newarray(t.bucket, 1) } }第二階段:尋找可插入位和更新既有值//根據低八位計算得到 bucket 的內存地址 bucket := hash & bucketMask(h.B) //判斷是否正在擴容,若正在擴容中則先遷移再接著處理 if h.growing() { growWork(t, h, bucket) } //計算并得到 bucket 的 bmap 指針地址 b := (*bmap)(unsafe.Pointer(uintptr(h.buckets) + bucket*uintptr(t.bucketsize))) //計算 key hash 高八位用于查找 Key top := tophash(hash) var inserti *uint8 var insertk unsafe.Pointer var val unsafe.Pointer for { //迭代 buckets 中的每一個 bucket(共 8 個) for i := uintptr(0); i < bucketCnt; i++ { //對比 bucket.tophash 與 top(高八位)是否一致 if b.tophash[i] != top { //若不一致,判斷是否為空槽 if b.tophash[i] == empty && inserti == nil { //有兩種情況,第一種是沒有插入過。第二種是插入后被刪除 inserti = &b.tophash[i] insertk = add(unsafe.Pointer(b), dataOffset+i*uintptr(t.keysize)) //把該位置標識為可插入 tophash 位置,這里就是第一個可以插入數據的地方 val = add(unsafe.Pointer(b), dataOffset+bucketCnt*uintptr(t.keysize)+i*uintptr(t.valuesize)) } continue } //若是匹配(也就是原本已經存在),則進行更新。最后跳出并返回 value 的內存地址 k := add(unsafe.Pointer(b), dataOffset+i*uintptr(t.keysize)) if t.indirectkey { k = *((*unsafe.Pointer)(k)) } if !alg.equal(key, k) { continue } // already have a mapping for key. Update it. if t.needkeyupdate { typedmemmove(t.key, k, key) } val = add(unsafe.Pointer(b), dataOffset+bucketCnt*uintptr(t.keysize)+i*uintptr(t.valuesize)) goto done } //判斷是否迭代完畢,若是則結束迭代 buckets 并更新當前桶位置 ovf := b.overflow(t) if ovf == nil { break } b = ovf } //若滿足三個條件:觸發最大 LoadFactor 、存在過多溢出桶 overflow buckets、沒有正在進行擴容。就會進行擴容動作(以確保后續的動作) if !h.growing() && (overLoadFactor(h.count+1, h.B) || tooManyOverflowBuckets(h.noverflow, h.B)) { hashGrow(t, h) goto again // Growing the table invalidates everything, so try again }第三階段:申請新的插入位和插入新值//經過前面迭代尋找動作,若沒有找到可插入的位置,意味著當前的所有桶都滿了,將重新分配一個新溢出桶用于插入動作。最后再在上一步申請的新插入位置,存儲鍵值對,返回該值的內存地址 if inserti == nil { // all current buckets are full, allocate a new one. newb := h.newoverflow(t, b) inserti = &newb.tophash[0] insertk = add(unsafe.Pointer(newb), dataOffset) val = add(insertk, bucketCnt*uintptr(t.keysize)) }// store new key/value at insert position if t.indirectkey { kmem := newobject(t.key) *(*unsafe.Pointer)(insertk) = kmem insertk = kmem } if t.indirectvalue { vmem := newobject(t.elem) *(*unsafe.Pointer)(val) = vmem } typedmemmove(t.key, insertk, key) *inserti = top h.count++done ... return val

第四階段:寫入

最后返回的是內存地址。是怎么進行寫入的呢?這是因為隱藏的最后一步寫入動作(將值拷貝到指定內存區域)是通過底層匯編配合來完成的,在 runtime 中只完成了絕大部分的動作。?mapassign?函數和拿到值存放的內存地址,再將 6666666 這個值存放進該內存地址中。另外我們看到?PCDATA?指令,主要是包含一些垃圾回收的信息,由編譯器產生。...0x0099 00153 (test.go:6) CALL runtime.mapassign_fast32(SB)0x009e 00158 (test.go:6) PCDATA $2, $20x009e 00158 (test.go:6) MOVQ 24(SP), AX0x00a3 00163 (test.go:6) PCDATA $2, $00x00a3 00163 (test.go:6) MOVL $6666666, (AX)

擴容:

關于上文中一直提到的擴容是怎么回事呢,現在我們來具體分析下:還記得bucket中的topHash字段嗎?現在我們來補充知識點minTopHash:當一個 cell 的 tophash 值小于 minTopHash 時,標志這個 cell 的遷移狀態。因為這個狀態值是放在 tophash 數組里,為了和正常的哈希值區分開,會給 key 計算出來的哈希值一個增量:minTopHash。這樣就能區分正常的 top hash 值和表示狀態的哈希值。下面的這幾種狀態就表征了 bucket 的情況:// 空的 cell,也是初始時 bucket 的狀態empty = 0// 空的 cell,表示 cell 已經被遷移到新的 bucketevacuatedEmpty = 1// key,value 已經搬遷完畢,但是 key 都在新 bucket 前半部分,evacuatedX = 2// 同上,key 在后半部分evacuatedY = 3// tophash 的最小正常值minTopHash = 4

為了避免計算出的topHash與minTopHash 沖突,底層做了相關操作:

func tophash(hash uintptr) uint8 { top := uint8(hash >> (sys.PtrSize*8 - 8)) if top < minTopHash { top += minTopHash } return top}隨著向 map 中添加的 key 越來越多,key 發生碰撞的概率也越來越大。bucket 中的 8 個 cell 會被逐漸塞滿,查找、插入、刪除 key 的效率也會越來越低。最理想的情況是一個 bucket 只裝一個 key,這樣,就能達到?O(1)的效率,但這樣空間消耗太大,用空間換時間的代價太高。Go 語言采用一個 bucket 里裝載 8 個 key,定位到某個 bucket 后,還需要再定位到具體的 key,這實際上又用了時間換空間。當然,這樣做,要有一個度,不然所有的 key 都落在了同一個 bucket 里,直接退化成了鏈表,各種操作的效率直接降為 O(n),是不行的。因此,需要有一個指標來衡量前面描述的情況,這就是?裝載因子。Go 源碼里這樣定義: loadFactor := count/(2^B) count 就是 map 的元素個數,2^B 表示 bucket 數量。再來說觸發 map 擴容的時機:在向 map 插入新 key 的時候,會進行條件檢測,符合下面這 2 個條件,就會觸發擴容:1、裝載因子超過閾值,源碼里定義的閾值是 6.52、overflow 的 bucket 數量過多通過匯編語言可以找到賦值操作對應源碼中的函數是?mapassign,對應擴容條件的源碼如下://觸發擴容的時機if !h.growing() && (overLoadFactor(h.count+1, h.B) || tooManyOverflowBuckets(h.noverflow, h.B)) { hashGrow(t, h) goto again // Growing the table invalidates everything, so try again }// 裝載因子超過 6.5func overLoadFactor(count int, B uint8) bool { return count > bucketCnt && uintptr(count) > loadFactorNum*(bucketShift(B)/loadFactorDen)}// overflow buckets 太多func tooManyOverflowBuckets(noverflow uint16, B uint8) bool { if B > 15 { B = 15 } return noverflow >= uint16(1)<}第 1 點:我們知道,每個 bucket 有 8 個空位,在沒有溢出,且所有的桶都裝滿了的情況下,裝載因子算出來的結果是 8。因此當裝載因子超過 6.5 時,表明很多 bucket 都快要裝滿了,查找效率和插入效率都變低了。在這個時候進行擴容是有必要的。第 2 點:是對第 1 點的補充。就是說在裝載因子比較小的情況下,這時候 map 的查找和插入效率也很低,而第 1 點識別不出來這種情況。表面現象就是計算裝載因子的分子比較小,即 map 里元素總數少,但是 bucket 數量多(真實分配的 bucket 數量多,包括大量的 overflow bucket)。不難想像造成這種情況的原因:不停地插入、刪除元素。先插入很多元素,導致創建了很多 bucket,但是裝載因子達不到第 1 點的臨界值,未觸發擴容來緩解這種情況。之后,刪除元素降低元素總數量,再插入很多元素,導致創建很多的 overflow bucket,但就是不會觸犯第 1 點的規定,你能拿我怎么辦?overflow bucket 數量太多,導致 key 會很分散,查找插入效率低得嚇人,因此出臺第 2 點規定。這就像是一座空城,房子很多,但是住戶很少,都分散了,找起人來很困難。對于命中條件 1,2 的限制,都會發生擴容。但是擴容的策略并不相同,畢竟兩種條件應對的場景不同。對于條件 1,元素太多,而 bucket 數量太少,很簡單:將 B 加 1,bucket 最大數量(2^B)直接變成原來 bucket 數量的 2 倍。于是,就有新老 bucket 了。注意,這時候元素都在老 bucket 里,還沒遷移到新的 bucket 來。新 bucket 只是最大數量變為原來最大數量的 2 倍(2^B*2) 。對于條件 2,其實元素沒那么多,但是 overflow bucket 數特別多,說明很多 bucket 都沒裝滿。解決辦法就是開辟一個新 bucket 空間,將老 bucket 中的元素移動到新 bucket,使得同一個 bucket 中的 key 排列地更緊密。這樣,原來,在 overflow bucket 中的 key 可以移動到 bucket 中來。結果是節省空間,提高 bucket 利用率,map 的查找和插入效率自然就會提升。由于 map 擴容需要將原有的 key/value 重新搬遷到新的內存地址,如果有大量的 key/value 需要搬遷,會非常影響性能。因此 Go map 的擴容采取了一種稱為“漸進式”的方式,原有的 key 并不會一次性搬遷完畢,每次最多只會搬遷 2 個 bucket。上面說的?hashGrow()函數實際上并沒有真正地“搬遷”,它只是分配好了新的 buckets,并將老的 buckets 掛到了 oldbuckets 字段上。真正搬遷 buckets 的動作在?growWork()函數中,而調用?growWork()函數的動作是在 mapassign 和 mapdelete 函數中。也就是插入或修改、刪除 key 的時候,都會嘗試進行搬遷 buckets 的工作。先檢查 oldbuckets 是否搬遷完畢,具體來說就是檢查 oldbuckets 是否為 nil。func hashGrow(t *maptype, h *hmap) { // B+1 相當于是原來 2 倍的空間 bigger := uint8(1) // 對應條件 2 if !overLoadFactor(h.count+1, h.B) { // 進行等量的內存擴容,所以 B 不變 bigger = 0 h.flags |= sameSizeGrow } // 將老 buckets 掛到 buckets 上 oldbuckets := h.buckets // 申請新的 buckets 空間 newbuckets, nextOverflow := makeBucketArray(t, h.B+bigger, nil) //先把 h.flags 中 iterator 和 oldIterator 對應位清 0 //如果 iterator 位為 1,把它轉接到 oldIterator 位,使得 oldIterator 標志位變成1 //可以理解為buckets 現在掛到了 oldBuckets 名下了,將對應的標志位也轉接過去 flags := h.flags &^ (iterator | oldIterator) if h.flags&iterator != 0 { flags |= oldIterator } // commit the grow (atomic wrt gc) h.B += bigger h.flags = flags h.oldbuckets = oldbuckets h.buckets = newbuckets // 搬遷進度為 0 h.nevacuate = 0 // overflow buckets 數為 0 h.noverflow = 0}

幾個標志位如下:

// 可能有迭代器使用 bucketsiterator = 1// 可能有迭代器使用 oldbucketsoldIterator = 2// 有協程正在向 map 中寫入 keyhashWriting = 4// 等量擴容(對應條件 2)sameSizeGrow = 8

再來看看真正執行搬遷工作的 growWork() 函數

func growWork(t *maptype, h *hmap, bucket uintptr) { // 搬遷正在使用的舊 bucket evacuate(t, h, bucket&h.oldbucketmask()) // 再搬遷一個 bucket,以加快搬遷進程 if h.growing() { evacuate(t, h, h.nevacuate) }}func (h *hmap) growing() bool { return h.oldbuckets != nil}

搬遷過程evacuate源碼:

type evacDst struct { b *bmap // 表示bucket 移動的目標地址 i int // 指向 x,y 中 key/val 的 index k unsafe.Pointer // 指向 x,y 中的 key v unsafe.Pointer // 指向 x,y 中的 value}func evacuate(t *maptype, h *hmap, oldbucket uintptr) { // 定位老的 bucket 地址 b := (*bmap)(add(h.oldbuckets, oldbucket*uintptr(t.bucketsize))) // 計算容量 結果是 2^B,如 B = 5,結果為32 newbit := h.noldbuckets() // 如果 b 沒有被搬遷過 if !evacuated(b) { // 默認是等 size 擴容,前后 bucket 序號不變 var xy [2]evacDst // 使用 x 來進行搬遷 x := &xy[0] x.b = (*bmap)(add(h.buckets, oldbucket*uintptr(t.bucketsize))) x.k = add(unsafe.Pointer(x.b), dataOffset) x.v = add(x.k, bucketCnt*uintptr(t.keysize)) // 如果不是等 size 擴容,前后 bucket 序號有變 if !h.sameSizeGrow() { // 使用 y 來進行搬遷 y := &xy[1] // y 代表的 bucket 序號增加了 2^B y.b = (*bmap)(add(h.buckets, (oldbucket+newbit)*uintptr(t.bucketsize))) y.k = add(unsafe.Pointer(y.b), dataOffset) y.v = add(y.k, bucketCnt*uintptr(t.keysize)) } // 遍歷所有的 bucket,包括 overflow buckets b 是老的 bucket 地址 for ; b != nil; b = b.overflow(t) { k := add(unsafe.Pointer(b), dataOffset) v := add(k, bucketCnt*uintptr(t.keysize)) // 遍歷 bucket 中的所有 cell for i := 0; i < bucketCnt; i, k, v = i+1, add(k, uintptr(t.keysize)), add(v, uintptr(t.valuesize)) { // 當前 cell 的 top hash 值 top := b.tophash[i] // 如果 cell 為空,即沒有 key if top == empty { // 那就標志它被"搬遷"過 b.tophash[i] = evacuatedEmpty continue } // 正常不會出現這種情況 // 未被搬遷的 cell 只可能是 empty 或是 // 正常的 top hash(大于 minTopHash) if top < minTopHash { throw("bad map state") } // 如果 key 是指針,則解引用 k2 := k if t.indirectkey { k2 = *((*unsafe.Pointer)(k2)) } var useY uint8 // 如果不是等量擴容 if !h.sameSizeGrow() { // 計算 hash 值,和 key 第一次寫入時一樣 hash := t.key.alg.hash(k2, uintptr(h.hash0)) // 如果有協程正在遍歷 map 如果出現 相同的 key 值,算出來的 hash 值不同 if h.flags&iterator != 0 && !t.reflexivekey && !t.key.alg.equal(k2, k2) { // useY =1 使用位置Y useY = top & 1 top = tophash(hash) } else { // 第 B 位置 不是 0 if hash&newbit != 0 { //使用位置Y useY = 1 } } } if evacuatedX+1 != evacuatedY { throw("bad evacuatedN") } //決定key是裂變到 X 還是 Y b.tophash[i] = evacuatedX + useY // evacuatedX + 1 == evacuatedY dst := &xy[useY] // evacuation destination // 如果 xi 等于 8,說明要溢出了 if dst.i == bucketCnt { // 新建一個 bucket dst.b = h.newoverflow(t, dst.b) // xi 從 0 開始計數 dst.i = 0 //key移動的位置 dst.k = add(unsafe.Pointer(dst.b), dataOffset) //value 移動的位置 dst.v = add(dst.k, bucketCnt*uintptr(t.keysize)) } // 設置 top hash 值 dst.b.tophash[dst.i&(bucketCnt-1)] = top // mask dst.i as an optimization, to avoid a bounds check // key 是指針 if t.indirectkey { // 將原 key(是指針)復制到新位置 *(*unsafe.Pointer)(dst.k) = k2 // copy pointer } else { // 將原 key(是值)復制到新位置 typedmemmove(t.key, dst.k, k) // copy value } //value同上 if t.indirectvalue { *(*unsafe.Pointer)(dst.v) = *(*unsafe.Pointer)(v) } else { typedmemmove(t.elem, dst.v, v) } // 定位到下一個 cell dst.i++ dst.k = add(dst.k, uintptr(t.keysize)) dst.v = add(dst.v, uintptr(t.valuesize)) } } // Unlink the overflow buckets & clear key/value to help GC. // bucket搬遷完畢 如果沒有協程在使用老的 buckets,就把老 buckets 清除掉,幫助gc if h.flags&oldIterator == 0 && t.bucket.kind&kindNoPointers == 0 { b := add(h.oldbuckets, oldbucket*uintptr(t.bucketsize)) ptr := add(b, dataOffset) n := uintptr(t.bucketsize) - dataOffset memclrHasPointers(ptr, n) } } // 更新搬遷進度 if oldbucket == h.nevacuate { advanceEvacuationMark(h, t, newbit) }}

擴容后,B 增加了 1,意味著 buckets 總數是原來的 2 倍,原來 1 號的桶“裂變”到兩個桶,某個 key 在搬遷前后 bucket 序號可能和原來相等,也可能是相比原來加上 2^B(原來的 B 值),取決于 hash 值 第 6 bit 位是 0 還是 1。原理看下圖:

4、遍歷操作:1.只獲取key for key := range m { fmt.Println(key) }2.只獲取value for _, value := range m { fmt.Println(value) }3.有序遍歷map,獲取kv keys := []string{} for k, _ := range m { keys = append(keys, k) } // 排序 sort.Strings(keys) // 有序遍歷 for _, k := range keys { fmt.Println(k, m[k]) }理解了上面 bucket 序號的變化,我們就可以回答另一個問題了:為什么遍歷 map 是無序的?遍歷的過程,就是按順序遍歷 bucket,同時按順序遍歷 bucket 中的 key。搬遷后,key 的位置發生了重大的變化,有些 key 飛上高枝,有些 key 則原地不動。這樣,遍歷 map 的結果就不可能按原來的順序了。當然,如果我就一個 hard code 的 map,我也不會向 map 進行插入刪除的操作,按理說每次遍歷這樣的 map 都會返回一個固定順序的 key/value 序列吧。的確是這樣,但是 Go 杜絕了這種做法,因為這樣會給新手程序員帶來誤解,以為這是一定會發生的事情,在某些情況下,可能會釀成大錯。當然,Go 做得更絕,當我們在遍歷 map 時,并不是固定地從 0 號 bucket 開始遍歷,每次都是從一個隨機值序號的 bucket 開始遍歷,并且是從這個 bucket 的一個隨機序號的 cell 開始遍歷。這樣,即使你是一個寫死的 map,僅僅只是遍歷它,也不太可能會返回一個固定序列的 key/value 對了。//runtime.mapiterinit 遍歷時選用初始桶的函數func mapiterinit(t *maptype, h *hmap, it *hiter) { ... it.t = t it.h = h it.B = h.B it.buckets = h.buckets if t.bucket.kind&kindNoPointers != 0 { h.createOverflow() it.overflow = h.extra.overflow it.oldoverflow = h.extra.oldoverflow } r := uintptr(fastrand()) if h.B > 31-bucketCntBits { r += uintptr(fastrand()) << 31 } it.startBucket = r & bucketMask(h.B) it.offset = uint8(r >> h.B & (bucketCnt - 1)) it.bucket = it.startBucket ... mapiternext(it)}

重點是fastrand?的部分,是一個生成隨機數的方法:它生成了隨機數。用于決定從哪里開始循環迭代。更具體的話就是根據隨機數,選擇一個桶位置作為起始點進行遍歷迭代因此每次重新?for?range?map,你見到的結果都是不一樣的。那是因為它的起始位置根本就不固定!

...// decide where to startr := uintptr(fastrand())if h.B > 31-bucketCntBits { r += uintptr(fastrand()) << 31}it.startBucket = r & bucketMask(h.B)it.offset = uint8(r >> h.B & (bucketCnt - 1))// iterator stateit.bucket = it.startBucket

5、更新操作:

底層操作原理參考上文

m["age"] = "two"m["name"] = "lily"6、刪除操作:delete(m, "name")寫操作底層的執行函數是?mapdelete:*func mapdelete(t *maptype, h?hmap, key unsafe.Pointer)它首先會檢查 h.flags 標志,如果發現寫標位是 1,直接 panic,因為這表明有其他協程同時在進行寫操作。計算 key 的哈希,找到落入的 bucket。檢查此 map 如果正在擴容的過程中,直接觸發一次搬遷操作。刪除操作同樣是兩層循環,核心還是找到 key 的具體位置。尋找過程都是類似的,在 bucket 中挨個 cell 尋找。找到對應位置后,對 key 或者 value 進行“清零”操作,將 count 值減 1,將對應位置的 tophash 值置成?Empty。func mapdelete(t *maptype, h *hmap, key unsafe.Pointer) { if raceenabled && h != nil { callerpc := getcallerpc() pc := funcPC(mapdelete) racewritepc(unsafe.Pointer(h), callerpc, pc) raceReadObjectPC(t.key, key, callerpc, pc) } if msanenabled && h != nil { msanread(key, t.key.size) } if h == nil || h.count == 0 { return } if h.flags&hashWriting != 0 { throw("concurrent map writes") } alg := t.key.alg hash := alg.hash(key, uintptr(h.hash0)) // Set hashWriting after calling alg.hash, since alg.hash may panic, // in which case we have not actually done a write (delete). h.flags |= hashWriting bucket := hash & bucketMask(h.B) if h.growing() { growWork(t, h, bucket) } b := (*bmap)(add(h.buckets, bucket*uintptr(t.bucketsize))) top := tophash(hash)search: for ; b != nil; b = b.overflow(t) { for i := uintptr(0); i < bucketCnt; i++ { if b.tophash[i] != top { continue } k := add(unsafe.Pointer(b), dataOffset+i*uintptr(t.keysize)) k2 := k if t.indirectkey { k2 = *((*unsafe.Pointer)(k2)) } if !alg.equal(key, k2) { continue } // Only clear key if there are pointers in it. // 對key清零 if t.indirectkey { *(*unsafe.Pointer)(k) = nil } else if t.key.kind&kindNoPointers == 0 { memclrHasPointers(k, t.key.size) } v := add(unsafe.Pointer(b), dataOffset+bucketCnt*uintptr(t.keysize)+i*uintptr(t.valuesize)) // 對value清零 if t.indirectvalue { *(*unsafe.Pointer)(v) = nil } else if t.elem.kind&kindNoPointers == 0 { memclrHasPointers(v, t.elem.size) } else { memclrNoHeapPointers(v, t.elem.size) } // 高位hash清零 b.tophash[i] = empty // 個數減一 h.count-- break search } } if h.flags&hashWriting == 0 { throw("concurrent map writes") } h.flags &^= hashWriting}

7、并發操作

map 并不是一個線程安全的數據結構。同時讀寫一個 map 是不安全的,如果被檢測到,會直接 panic。解決方法1:讀寫鎖?sync.RWMutex。type TestMap struct { M map[int]string Lock sync.RWMutex}func main() { testMap := TestMap{} testMap.M = map[int]string{1: "lili"} go func() { i := 0 for i < 10000 { testMap.Lock.RLock() fmt.Println(i, testMap.M[1]) testMap.Lock.RUnlock() i++ } }() go func() { i := 0 for i < 10000 { testMap.Lock.Lock() testMap.M[1] = "lily" testMap.Lock.Unlock() i++ } }() for { runtime.GC() }}解決方法2:使用golang提供的 sync.Mapfunc main() { m := sync.Map{} m.Store(1, 1) i := 0 go func() { for i < 1000 { m.Store(1, 1) i++ } }() go func() { for i < 1000 { m.Store(2, 2) i++ } }() go func() { for i < 1000 { fmt.Println(m.Load(1)) i++ } }() for { runtime.GC() }}參考文獻:

【1】《深度解密Go語言之map》

【2】《解剖Go語言map底層實現》

【3】《深入理解 Go map:賦值和擴容遷移》

Golang源碼系列會有持續性的文章發布,后續會在本公眾號陸續推出,感興趣的伙伴們敬請期待呦!

總結

以上是生活随笔為你收集整理的c++ map 获取key列表_好未来Golang源码系列一:Map实现原理分析的全部內容,希望文章能夠幫你解決所遇到的問題。

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

国产中文字幕视频在线观看 | 天天综合网久久 | 91桃色在线免费观看 | 成人av在线网 | 91视频黄色 | 久久毛片视频 | 91理论电影 | 欧美伦理一区二区三区 | 国产欧美日韩一区 | www·22com天天操 | 久草www | 国产色拍拍拍拍在线精品 | 国产美女精品视频免费观看 | 免费观看午夜视频 | av一级片网站 | 久久人人爽爽人人爽人人片av | 久久久精品国产一区二区三区 | 日韩乱理 | 99re久久精品国产 | 免费视频成人 | 黄色免费在线视频 | 欧美日韩xx | 亚洲欧美国产日韩在线观看 | 最近更新的中文字幕 | 久艹在线免费观看 | 日韩欧美aaa| 玖玖精品视频 | 国产小视频国产精品 | 日韩免费一区 | 五月天激情视频在线观看 | 三级黄色片在线观看 | 免费观看视频的网站 | 91免费的视频在线播放 | 中文字幕婷婷 | 97碰在线 | 国产一区二区在线免费观看 | 亚州精品天堂中文字幕 | 久久99国产精品免费网站 | 国产91精品久久久久久 | 精品一区二区三区香蕉蜜桃 | 日韩精品资源 | 午夜在线国产 | 国产女人免费看a级丨片 | 久久第四色 | 超碰在线资源 | 99草视频| av免费看网站 | 伊人久久av| 国内精品久久久久影院日本资源 | 欧美日韩不卡一区 | 亚洲黄色大片 | 欧美成人h版 | 久草在线观| www.天天成人国产电影 | 国产精品成人免费精品自在线观看 | 日日干av | 久久字幕网 | 人人爽人人干 | 96超碰在线 | 国产综合精品一区二区三区 | 中文av影院| 久久精品日本啪啪涩涩 | 国产午夜精品一区 | 国产老太婆免费交性大片 | 亚洲自拍偷拍色图 | 精品国精品自拍自在线 | 中文字幕视频 | 亚洲激情在线播放 | 国产精品国产亚洲精品看不卡15 | 国产69精品久久99不卡的观看体验 | 国产在线观看91 | 一级免费黄视频 | 亚洲天堂色婷婷 | 2019中文在线观看 | 欧洲激情综合 | 二区视频在线 | 精品国产久 | 国产一级淫片在线观看 | 免费av网站观看 | 九九热视频在线免费观看 | 亚洲精品国产自产拍在线观看 | 中文字幕在线中文 | 国产精品一区二区中文字幕 | 久草久草在线观看 | 国产精品免费视频网站 | 色婷婷久久 | 天天草天天 | 99久久精品国产毛片 | 久久成人欧美 | 久久资源在线 | 久久国产精品99久久久久久进口 | 久产久精国产品 | 97国产精品亚洲精品 | av网在线观看 | 亚洲视频,欧洲视频 | 婷婷丁香在线视频 | 日日夜夜精品视频天天综合网 | 日韩精品一区电影 | 久久视频免费在线 | www.久久成人 | 国产一区免费在线 | 91漂亮少妇露脸在线播放 | 人人添人人澡人人澡人人人爽 | 在线观看亚洲国产 | 国产精品毛片久久久久久 | 97在线播放| 久操视频在线播放 | 欧洲精品视频一区 | 亚洲免费在线观看视频 | 日韩在线三区 | 亚洲激情影院 | 婷婷电影在线观看 | 狠狠的干| 久草视频在线新免费 | 91视频在线观看免费 | 久久久精品国产免费观看一区二区 | 一二三久久久 | 久久看看 | 天天射天天射 | 色a在线观看 | 国产精品久久久久久久久久久免费看 | 久久天天躁夜夜躁狠狠躁2022 | 亚洲 在线| 欧美一二区在线 | 夜夜躁狠狠躁日日躁视频黑人 | 国产精品毛片一区视频播 | 国产福利一区二区在线 | www.国产在线 | 精品欧美在线视频 | 国产女人18毛片水真多18精品 | 一区二区三区福利 | 久久无码精品一区二区三区 | 国产欧美日韩精品一区二区免费 | 国产精品久久久久久久久久 | 欧美婷婷色| 久久国产精品99国产精 | 美女福利视频在线 | 国产精品99久久久久久久久久久久 | 五月激情六月丁香 | 国产精品一区二区在线看 | 国产91大片 | 日韩另类在线 | 国产一区在线观看视频 | 成人日批视频 | 久久精品牌麻豆国产大山 | 久久精品福利视频 | 欧美日韩免费观看一区=区三区 | 亚洲美女精品视频 | 在线播放亚洲激情 | 天天干国产 | 精品一二三区 | 特级aaa毛片 | 国产精品亚洲片夜色在线 | 日韩一区二区三区不卡 | 久久久久国产精品一区二区 | 99精品视频观看 | 国产在线久久久 | 国产精品久久久久久久久久久免费看 | 黄色大片日本免费大片 | 国产亚洲成av人片在线观看桃 | 免费亚洲黄色 | 国产精品久久99综合免费观看尤物 | 黄色a视频 | 久久免费视频这里只有精品 | 色综合色综合久久综合频道88 | 99精品热视频只有精品10 | 欧美日韩一区二区三区免费视频 | 激情婷婷在线 | 亚洲人久久久 | 国产精品6 | 在线观看中文字幕 | 国产精品99蜜臀久久不卡二区 | 欧洲激情在线 | 六月丁香激情综合色啪小说 | 久久美女电影 | 在线免费观看黄色小说 | 国产成人精品在线播放 | 免费网站观看www在线观看 | 欧美日韩后 | 亚洲激情五月 | 亚洲资源一区 | 一级欧美日韩 | 亚洲国产高清在线观看视频 | 极品久久久久久久 | 久久亚洲免费 | 色婷婷久久久综合中文字幕 | 天天操天天吃 | 中文字幕在线观看免费高清完整版 | 日韩天堂在线观看 | 中文字幕在线观看视频免费 | 日韩欧美视频在线免费观看 | 亚洲精品美女免费 | 国产黄色精品在线 | 精品在线观看免费 | 国产真实精品久久二三区 | 亚洲精品在线观看免费 | 久久久五月天 | 天天摸天天干天天操天天射 | 91精品办公室少妇高潮对白 | 99热手机在线 | 8090yy亚洲精品久久 | 国产一区二区三区久久久 | 婷婷午夜 | 亚洲一级二级 | 国产成人精品午夜在线播放 | 免费在线观看日韩视频 | www.com在线观看 | 欧美激情第十页 | 亚洲一区视频免费观看 | 97夜夜澡人人双人人人喊 | 综合成人在线 | 亚洲精品视频中文字幕 | 蜜臀aⅴ国产精品久久久国产 | 久久9精品 | 婷婷网站天天婷婷网站 | 精品在线视频一区二区三区 | 4438全国亚洲精品在线观看视频 | 玖玖玖在线观看 | 999精品| 欧美成人xxx | 日韩免费成人av | 免费av小说| 欧美性色综合网 | 国产精品成人品 | 国产精品一级在线 | 91视频首页 | 国产精品久久久久久久久久妇女 | 久久成人午夜 | 在线观看视频色 | 黄色网www | 最新99热 | 久久怡红院 | 在线v| 夜夜婷婷| 欧美一区二区在线免费观看 | 欧美日性视频 | 亚洲女欲精品久久久久久久18 | 91精品国产自产在线观看永久 | 在线你懂 | 在线日韩中文字幕 | 69视频在线播放 | 狠狠色丁香婷婷综合基地 | 久久视频免费在线 | 色婷av| 亚洲麻豆精品 | 精品国产人成亚洲区 | 久久久噜噜噜久久久 | 丁香综合激情 | 91少妇精拍在线播放 | 91精品国产99久久久久久久 | 日韩av一卡二卡三卡 | 国产三级在线播放 | 99精品国产亚洲 | www.亚洲视频| 天天狠狠 | 久久久久久久久免费视频 | 国产视频在线观看免费 | 日韩在线视频线视频免费网站 | 992tv在线| 亚洲最大色| 日本电影久久 | 激情欧美一区二区免费视频 | 日韩videos高潮hd | 在线观看免费一区 | 超碰97人| 久久久久久久久网站 | 毛片a级片| 狠狠色噜噜狠狠狠狠 | 中文字幕欧美日韩va免费视频 | 国产国产人免费人成免费视频 | 免费色网站 | av免费在线网 | 日韩免费一级电影 | 人人玩人人弄 | 99在线高清视频在线播放 | 国产成人一区二区啪在线观看 | 精品久久久久久亚洲 | 五月婷婷婷婷婷 | 亚洲第二色 | 不卡精品 | 激情丁香 | 日韩免费一区 | 五月激情姐姐 | 久久久噜噜噜久久久 | 欧美日韩一区二区三区免费视频 | 日本中文字幕影院 | 免费日韩电影 | 欧美在线aaa | 一级片黄色片网站 | 久久综合久久久久88 | 久久综合久色欧美综合狠狠 | 日韩综合精品 | 天天操天天添 | 97超碰在线人人 | 欧美激情第一区 | 黄色片网站免费 | 亚洲第一香蕉视频 | 久久综合九色欧美综合狠狠 | 一区二区中文字幕在线 | 天天色天天射天天操 | 成人免费在线网 | 99免在线观看免费视频高清 | 国产成人精品不卡 | 国产中年夫妇高潮精品视频 | 欧美日本一二三 | 天天鲁天天干天天射 | 欧美日韩在线观看一区 | 婷婷丁香狠狠爱 | 精品久久久久久久久久岛国gif | 成人在线观看你懂的 | 国产精品岛国久久久久久久久红粉 | 国产a高清| 五月黄色 | 久草视频在线资源 | 91成人黄色 | 91在线播放视频 | 91国内产香蕉 | 国产精品色在线 | 一区三区视频 | 国产精品资源网 | 色婷婷激情五月 | aaa亚洲精品一二三区 | 成人av免费在线 | 久久免费的视频 | 亚洲精品国精品久久99热 | 98精品国产自产在线观看 | av在线免费在线 | 操久 | 国产日韩精品一区二区三区 | 最近免费中文字幕 | 色婷婷免费视频 | 91片在线观看 | 中文字幕一区二区三区乱码不卡 | 国产美女网 | 欧美日韩aa | 久久人人爽人人片 | 国产精品资源网 | 欧美另类高潮 | www在线免费观看 | 国产a国产 | 中文字幕一区二区三区四区 | 婷婷伊人五月天 | 四虎视频| 黄色大片免费播放 | 极品美女被弄高潮视频网站 | 日本精品一区二区三区在线播放视频 | 欧美日韩精品在线观看视频 | www.狠狠操 | 国产精品黄色影片导航在线观看 | 中文字幕在线免费97 | 久一久久 | 日本激情动作片免费看 | 天天操夜夜操国产精品 | 久久亚洲国产精品 | 在线播放视频一区 | 又黄又爽又色无遮挡免费 | 99综合电影在线视频 | 欧美激情综合五月色丁香 | 日韩最新中文字幕 | 综合网天天 | 免费久久久久久 | 黄视频色网站 | 久久久www成人免费毛片麻豆 | 九九国产精品视频 | 超碰免费av | 成人久久久电影 | 欧美在线资源 | 亚洲精品动漫成人3d无尽在线 | 久久视屏网 | av大全在线免费观看 | 日韩乱色精品一区二区 | 麻豆一二三精选视频 | 在线观看免费av网站 | 99精品影视 | 99在线精品视频 | 国产精品自在欧美一区 | 狠狠色2019综合网 | 亚洲精品视频中文字幕 | 天天射天天 | 久久免费福利视频 | 狠狠干电影 | 欧美久久99 | 永久免费毛片在线观看 | 日韩a欧美 | 日韩欧美69| 精品国产自在精品国产精野外直播 | 久久视频国产精品免费视频在线 | 日韩精品一区二区三区丰满 | 91香蕉视频色版 | 久久精品草 | 97超碰人人澡人人爱学生 | 韩日三级在线 | 欧美男男tv网站 | 国产精品日韩在线播放 | 欧美成人xxxxx | 粉嫩一区二区三区粉嫩91 | 久久久久免费精品国产 | 日韩精品最新在线观看 | 成人小视频在线观看免费 | 国产成人一区二区三区 | 亚洲人成网站精品片在线观看 | 成人国产一区二区 | 国产色在线 | 97人人添人澡人人爽超碰动图 | 亚洲乱码在线观看 | 国产精品久久一区二区无卡 | 国产精品第一视频 | 九色精品| 91福利社在线观看 | a色网站| 热久久影视| 久久人人97超碰精品888 | 国产成人久久精品77777综合 | 日韩极品视频在线观看 | 亚洲一区久久久 | 国产亚洲精品日韩在线tv黄 | 免费视频你懂得 | 精壮的侍卫呻吟h | 欧美一区三区四区 | 热久久在线视频 | 亚洲激情电影在线 | 免费中文字幕在线观看 | 麻豆成人小视频 | 久久人人爽人人人人片 | 激情欧美一区二区免费视频 | 成人黄性视频 | 中文字幕精品三级久久久 | 午夜精品视频在线 | 国产午夜精品福利视频 | 国产黄大片在线观看 | 国产成人av网址 | 国产精品久久久久一区二区国产 | 久草综合在线观看 | www.久久91 | 日韩视频一区二区三区在线播放免费观看 | 国产欧美日韩视频 | 亚洲综合色视频在线观看 | 国产视频69| 国产黄色在线观看 | 亚洲伦理一区 | 91视频高清完整版 | 亚洲激情免费 | 天天干天天操天天做 | 天天操夜夜逼 | 色婷婷激情四射 | 欧美极品xxxx| 九色视频网站 | 国产91精品一区二区绿帽 | 在线播放视频一区 | 深夜免费小视频 | 国产精品二区在线观看 | 看片一区二区三区 | 精品国产欧美一区二区三区不卡 | 久久国产片 | 四虎国产精品免费观看视频优播 | 国产黄影院色大全免费 | 国产成在线观看免费视频 | 国产精品第7页 | 韩国三级在线一区 | 精品国产乱码久久久久久1区二区 | 日本久久久久久久久 | 国产小视频精品 | a'aaa级片在线观看 | 激情影音先锋 | 四虎影视成人永久免费观看视频 | 欧美一区二区三区激情视频 | 天天操天天爽天天干 | 久久精品激情 | 亚洲色综合 | 亚洲 欧美 另类人妖 | 欧美在线观看视频 | 国产黑丝一区二区 | www.久草视频 | 摸bbb搡bbb搡bbbb| 国产精品一区二区果冻传媒 | 久久国产精品99国产 | 天天综合亚洲 | 在线成人免费电影 | 久久网站最新地址 | 欧美韩国日本在线 | av成人免费在线看 | 久久黄色精品视频 | 欧美成人精品欧美一级乱 | 亚洲专区欧美 | 狠狠色丁香婷婷综合久小说久 | 日韩av中文字幕在线免费观看 | 久久综合精品国产一区二区三区 | 日韩在线电影观看 | 91亚·色| 国产黄色在线 | 在线看片成人 | 91午夜精品 | 日韩欧美视频在线观看免费 | 综合婷婷丁香 | 国内精品久久久久久久久 | 亚洲精品国产片 | 丰满少妇对白在线偷拍 | 美女黄频视频大全 | 久久久久久久久久影视 | 国产中文字幕91 | 综合激情网...| 久久伊人八月婷婷综合激情 | 欧美男同视频网站 | 精品视频中文字幕 | 国产亚洲精品成人 | 99精品在线视频播放 | 国产精品久久综合 | 国产精品欧美久久 | 香蕉在线视频观看 | 丁香五月亚洲综合在线 | 伊人亚洲综合 | 伊人婷婷激情 | 日韩一区正在播放 | 在线观看av免费 | 日韩精品免费一区二区 | 黄色日本免费 | 天天色综合久久 | 日韩在线精品 | 亚洲国产高清在线观看视频 | 全黄网站 | 99 久久久久| 国产高清精品在线 | 在线观看免费91 | 天天色婷婷 | 天天射天天色天天干 | 日韩精品中文字幕久久臀 | 欧美中文字幕第一页 | 天天艹天天干天天 | 婷婷在线色 | 日本最大色倩网站www | 亚洲精品视频一 | 亚洲aⅴ一区二区三区 | www.天天成人国产电影 | 国产成人精品一二三区 | 精品国产欧美一区二区 | 美女视频a美女大全免费下载蜜臀 | av一级一片| 欧美黑人巨大xxxxx | 最近中文字幕大全中文字幕免费 | 成年人网站免费在线观看 | 国产麻豆精品久久一二三 | 日韩免费福利 | 日韩三级视频在线观看 | 欧美 亚洲 另类 激情 另类 | 婷婷六月综合亚洲 | 久久国产精品99国产 | 亚洲一区不卡视频 | 在线观看日韩国产 | 国产精品女主播一区二区三区 | 成人欧美一区二区三区黑人麻豆 | 国产精品毛片久久蜜 | 国产精品麻豆果冻传媒在线播放 | www.色综合.com | 日韩精品一区二区在线视频 | 草草草影院 | 在线视频久 | 亚洲一一在线 | 欧美激情视频一区二区三区免费 | 一区二区中文字幕在线 | 天天操天天干天天操天天干 | 亚洲一区二区精品视频 | 国产精品久久久久av福利动漫 | 国产超碰在线观看 | 欧美另类性| 国产黄色片免费在线观看 | 久久免费片| 国产亚洲精品久久久久久大师 | 久久久久久久久久久精 | 人人爽人人爽人人片av | 在线看日韩av | 日韩性片 | 久久久网站 | 日韩精品一区二区三区三炮视频 | 五月天中文字幕mv在线 | 亚洲激情在线播放 | 超碰97在线看 | 奇米影视在线99精品 | 亚洲mv大片欧洲mv大片免费 | 五月天丁香视频 | 国产黄色播放 | 久久久国产精品麻豆 | 国产精品乱码一区二三区 | 黄色毛片一级片 | 天天干天天摸天天操 | 国产精品99久久久久的智能播放 | 久久久精品一区二区 | 在线观看国产www | 午夜成人免费电影 | 视频在线日韩 | 精品你懂的 | 97成人精品 | 精品欧美一区二区精品久久 | 99久久国产免费,99久久国产免费大片 | 一二三区视频在线 | 91亚洲精品在线观看 | 亚洲人成免费 | 日韩在线观看网址 | 精品一区二区av | 91亚洲精品国偷拍自产在线观看 | 亚洲人成网站精品片在线观看 | 午夜精品久久久久久 | av动图| 在线免费观看涩涩 | 97超级碰碰碰碰久久久久 | 中文超碰字幕 | 黄色软件在线看 | 中文字幕色播 | 91精品国产乱码久久 | 一本之道乱码区 | 色av色av色av | 日韩久久久久久久久久久久 | 国产偷v国产偷∨精品视频 在线草 | 日韩av不卡播放 | 一区二区三区在线免费播放 | 国产精品女同一区二区三区久久夜 | 人人狠狠综合久久亚洲婷 | 91av在线看 | 91成人短视频在线观看 | 婷婷丁香色综合狠狠色 | 欧美大片在线看免费观看 | 欧美激情综合五月色丁香 | 中文字幕在线专区 | 超碰人人射 | 久草在线视频首页 | 99久久99久久 | 欧美网站黄色 | 日韩手机在线观看 | 久久精品高清视频 | 狂野欧美激情性xxxx | 欧美成人精品欧美一级乱黄 | 亚洲黄色免费在线 | 亚洲高清视频在线观看免费 | 日韩精品视频久久 | 欧美伦理一区二区三区 | 韩日成人av | 久久久三级视频 | 国产精品久久久久久久久久ktv | 久久综合精品一区 | 欧美成人xxx | 午夜视频欧美 | 国产日产在线观看 | 久久国产视屏 | 欧美激情综合网 | 色福利网| 精品国产免费看 | 播五月综合| 国产午夜精品一区二区三区 | 一区二区视频电影在线观看 | 丝袜美女在线观看 | 亚洲另类人人澡 | 91人人爱 | 日韩色中色 | 久久精品99国产精品日本 | 热99久久精品 | 丁香5月婷婷久久 | av在线播放观看 | 亚洲在线看| 亚洲三级毛片 | 国产精品网红直播 | 麻豆一区二区三区视频 | 中文字幕色婷婷在线视频 | 中文字幕精品一区久久久久 | 免费在线看v | 国产高清福利在线 | 一级黄色片在线免费看 | 亚洲精品成人av在线 | 免费一级片在线 | 婷婷在线播放 | 精品一区久久 | 99精品成人 | 国产精品久久久久久超碰 | 美女久久久久久久久久 | 亚洲精品一区二区18漫画 | 国产精品久久久久久一区二区 | av天天澡天天爽天天av | 蜜臀av夜夜澡人人爽人人桃色 | 中文字幕高清在线播放 | 91精品国产91久久久久久三级 | 精品一区二区在线免费观看 | 麻豆小视频在线观看 | 国产黄av | 在线观看日韩一区 | 五月天网站在线 | 欧美极品少妇xxxxⅹ欧美极品少妇xxxx亚洲精品 | 亚洲欧美日韩一二三区 | 就色干综合 | 国产视频第二页 | 国产一线二线三线在线观看 | 国产成人久久精品亚洲 | 久久婷婷丁香 | 国产精品综合久久久久久 | 日韩av电影免费观看 | a在线播放 | av成人在线网站 | 国产三级午夜理伦三级 | 国产一级性生活 | 日韩字幕在线观看 | 国产最新精品视频 | 黄色小说网站在线 | 成人av影视在线 | 日日爽天天 | 久久综合久久综合九色 | 五月婷香 | 国产人成看黄久久久久久久久 | 蜜臀aⅴ精品一区二区三区 久久视屏网 | 美女网站黄免费 | 日韩欧美视频一区二区 | 日韩欧美在线综合网 | 一区二区三区高清在线 | 亚洲国产视频在线 | 最新av电影网站 | 97视频免费观看2区 亚洲视屏 | 中文字幕成人 | 99精品国产一区二区三区不卡 | 激情久久一区二区三区 | 九九热精品在线 | 国产精品美女久久久 | 国产精品成久久久久三级 | 亚洲精品乱码久久久久久写真 | 国产亚洲一区二区在线观看 | 夜夜嗨av色一区二区不卡 | 久久视频 | 精品国产免费av | 国产人成一区二区三区影院 | 又黄又爽免费视频 | 在线性视频日韩欧美 | 精品一区二区精品 | 久久国产系列 | 亚洲午夜av | 一本一本久久a久久精品综合 | 亚洲精品日韩一区二区电影 | 五月婷婷欧美视频 | 中文字幕免费看 | 亚洲精品乱码久久久久久高潮 | 四虎在线免费观看 | 日韩三级.com | 久久久18 | 中中文字幕av在线 | 日韩电影中文字幕在线观看 | 婷婷中文字幕在线观看 | 国产免费久久av | www.久久久.cum| 99热999| 国产福利在线 | 一区二区三区免费看 | 国产精品久久久区三区天天噜 | 涩涩伊人 | 久久99精品久久只有精品 | 在线观看日韩一区 | 91在线精品视频 | 国产精品美女久久久免费 | 久久久久久久久久久久99 | 久久久久国产成人精品亚洲午夜 | 国产女教师精品久久av | 91桃色免费观看 | 麻豆高清免费国产一区 | 亚洲欧洲av在线 | 欧美在线观看小视频 | 久久的色 | 久久激五月天综合精品 | 五月激情丁香 | 成人av在线直播 | 中文字幕免费观看 | 91香蕉亚洲精品 | 成年人天堂com | 又粗又长又大又爽又黄少妇毛片 | 99热999 | 婷婷天天色 | 欧美激情综合网 | 在线视频a | 亚欧日韩av| 亚洲激情视频在线 | 日韩videos高潮hd | 亚洲在线国产 | 色视频在线观看 | 天天摸日日操 | 国产91综合一区在线观看 | 欧美三级免费 | 国产专区一 | 色午夜| 黄av在线| www色网站| www.午夜 | av东方在线 | 黄色免费大全 | 一级特黄av | 日韩高清无线码2023 | 久久精品福利视频 | 色婷婷亚洲 | 国产亚州av | 91中文字幕永久在线 | 日本不卡一区二区三区在线观看 | 国产日韩欧美在线一区 | 亚洲国产精品久久久久久 | 99日精品 | 91久草视频 | 亚洲三级在线免费观看 | 黄色免费在线看 | 日韩av午夜 | 日韩在线观看视频中文字幕 | 国产精品在线看 | 五月天国产精品 | 欧美色图另类 | 一区二区激情视频 | 久草在线高清视频 | 国产一级电影网 | 国产高清在线免费视频 | 国产在线观看不卡 | 久久69精品久久久久久久电影好 | 91 在线视频播放 | 精品国产一区二 | 天天综合网 天天综合色 | 国产一性一爱一乱一交 | 婷婷黄色片 | 九九欧美视频 | 99草视频在线观看 | 二区三区av | 国产又粗又猛又黄又爽的视频 | 久久久久久久久免费视频 | 久艹在线观看视频 | 久久精品国产免费看久久精品 | 国产无遮挡猛进猛出免费软件 | 亚洲视频高清 | 91精品国产综合久久久久久久 | 五月激情亚洲 | 伊人资源站 | 国产中文字幕在线播放 | 99tvdz@gmail.com | 日韩一区二区三区在线观看 | 日韩精品视频免费专区在线播放 | 四虎在线观看视频 | 婷婷在线免费视频 | 国产久草在线观看 | 婷婷综合成人 | 国产成人精品一区二区三区网站观看 | 久久久 激情 | 亚洲欧美日本一区二区三区 | 国产精品福利视频 | 天天综合入口 | 天天爱综合 | 亚洲黄色片在线 | 在线观看视频免费播放 | 黄污网站在线 | 久久久久久久国产精品视频 | 国产色拍拍拍拍在线精品 | 日韩黄色在线观看 | 国产黄在线 | 国产在线播放一区 | 一级免费av | 国产精品嫩草影院123 | 伊人国产女 | 精品色999| 又黄又爽又色无遮挡免费 | 色婷婷综合久久久 | 久草免费在线视频 | 婷婷在线不卡 | 黄色大片日本免费大片 | 综合色站导航 | 香蕉网站在线观看 | 中文字幕免费高清 | 超碰国产在线观看 | av免费网站在线观看 | 中文字幕在线播放av | 美女黄频视频大全 | 91精品国产91p65| 国产97在线视频 | 日韩精品一区二区三区水蜜桃 | 久久久久亚洲最大xxxx | 99久久精品国产亚洲 | 亚洲狠狠操| 免费观看成年人视频 | 国产又黄又爽无遮挡 | 99热精品国产| 成人国产精品一区二区 | 99在线精品视频在线观看 | 成人国产精品入口 | 337p日本大胆噜噜噜噜 | 99视频在线免费看 | 视频在线91 | 91最新在线视频 | 三级性生活视频 | 国产不卡精品 | 免费精品久久久 | 伊人资源视频在线 | 国产精品久久电影观看 | 精品久久久久久久久亚洲 | 免费在线观看黄网站 | .精品久久久麻豆国产精品 亚洲va欧美 | 国产日韩精品在线 | 国产免费小视频 | 天天爱天天草 | 天天撸夜夜操 | 国产va饥渴难耐女保洁员在线观看 | 日韩中文字幕第一页 | 在线播放你懂 | 操操日日| 国产精品成 | 中文字幕在线观看91 | 国产精品欧美一区二区 | 97精品国产一二三产区 | 国产精品尤物视频 | 国产电影黄色av | 麻豆成人网 | 操一草| 99产精品成人啪免费网站 | 成人一区不卡 | 日韩一二区在线 | 欧美精品久久人人躁人人爽 | 欧美伦理电影一区二区 | 99这里只有 | 丁香在线观看完整电影视频 | 国产精品入口66mio女同 | 91精品免费在线观看 | 欧美一进一出抽搐大尺度视频 | 亚洲黄色在线免费观看 | 超碰在线cao| 奇米影视四色8888 | 精品免费视频 | 国产无套一区二区三区久久 | 91精品国产欧美一区二区成人 | 探花国产在线 | 久久精品xxx | 国产不卡在线观看视频 | 亚洲精品系列 | 色资源中文字幕 | 亚洲经典中文字幕 | 亚洲综合成人婷婷小说 | 99久久精品日本一区二区免费 | 久久影视一区 | av电影在线观看完整版一区二区 | 婷婷久久国产 | 99在线播放 | 国产精品久久99 | www免费看 | 99产精品成人啪免费网站 | 久久不卡免费视频 | 日韩精品影视 | 亚洲精品66 | 91成熟丰满女人少妇 | 国产精品久久久久婷婷二区次 | 91网站观看 | 国产精品美女久久久久久久久久久 | 一区二区三区在线观看免费视频 | 国产精品中文在线 | 国产美女黄网站免费 | 中文字幕在线精品 | 亚洲黄a| 手机av在线网站 | 天天综合网天天综合色 | 人人澡人人草 | 久久国语露脸国产精品电影 | 色婷婷综合在线 | 国产一区二区综合 | 超碰97在线人人 | 五月天久久久久久 | 日韩激情视频在线观看 | 狠狠做六月爱婷婷综合aⅴ 日本高清免费中文字幕 | 亚洲精品乱码久久久久久写真 | 欧美日韩国产一区二区三区在线观看 | 亚洲国产精品传媒在线观看 | 国产在线一区二区三区播放 | 久久一区国产 | 在线观看视频国产 | 色婷婷免费视频 | 免费在线观看91 | 日韩一区正在播放 | 亚洲成成品网站 | 18久久久 | 97在线观看免费高清 | 日韩欧美国产精品 | 婷婷久久一区二区三区 | 香蕉影院在线观看 | www.色五月.com| 天天夜夜狠狠操 | 亚洲少妇影院 | 小草av在线播放 | 日韩在线视频国产 | 久久久久电影网站 | 国产成人精品亚洲a | av电影亚洲| 国产精品一区二区62 | www.神马久久 | 国产一级在线观看视频 | 日韩字幕在线观看 | 免费在线色视频 | 草久在线视频 | www欧美xxxx | 亚洲免费视频在线观看 | 国产中文a | 日韩一区二区三区在线观看 | 国产中文字幕久久 | 国产精品久久久久久久久久三级 | 中文字幕在线观 | 五月婷婷在线综合 | 射九九| 日韩av电影网站在线观看 | 免费看搞黄视频网站 | 免费av 在线 |