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

歡迎訪問 生活随笔!

生活随笔

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

编程问答

深入理解Spark 2.1 Core (十):Shuffle Map 端的原理与源码分析

發布時間:2024/1/23 编程问答 32 豆豆
生活随笔 收集整理的這篇文章主要介紹了 深入理解Spark 2.1 Core (十):Shuffle Map 端的原理与源码分析 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
在上一篇《深入理解Spark 2.1 Core (九):迭代計算和Shuffle的原理與源碼分析》提到經過迭代計算后, SortShuffleWriter.write中: // 根據排序方式,對數據進行排序并寫入內存緩沖區。// 若排序中計算結果超出的閾值,// 則將其溢寫到磁盤數據文件sorter.insertAll(records)
  • 1

我們先來宏觀的了解下Map端,我們會根據aggregator.isDefined是否定義了聚合函數和ordering.isDefined是否定義了排序函數分為三種:

  • 沒有聚合和排序,數據先按照partition寫入不同的文件中,最后按partition順序合并寫入同一文件 。適合partition數量較少時。將多個bucket合并到同一文件,減少map輸出文件數,節省磁盤I/O,提高性能。
  • 沒有聚合但有排序,在緩存對數據先根據分區(或者還有key)進行排序,最后按partition順序合并寫入同一文件。適合當partition數量較多時。將多個bucket合并到同一文件,減少map輸出文件數,節省磁盤I/O,提高性能。緩存使用超過閾值,將數據寫入磁盤。
  • 有聚合有排序,現在緩存中根據key值聚合,再在緩存對數據先根據分區(或者還有key)進行排序,最后按partition順序合并寫入同一文件。將多個bucket合并到同一文件,減少map輸出文件數,節省磁盤I/O,提高性能。緩存使用超過閾值,將數據寫入磁盤。逐條的讀取數據,并進行聚合,減少了內存的占用。

我們先來深入看下insertAll:

def insertAll(records: Iterator[Product2[K, V]]): Unit = {// 若定義了聚合函數,則shouldCombine為trueval shouldCombine = aggregator.isDefined// 外部排序是否需要聚合if (shouldCombine) { // mergeValue 是 對 Value 進行 merge的函數val mergeValue = aggregator.get.mergeValue// createCombiner 為生成 Combiner 的 函數val createCombiner = aggregator.get.createCombinervar kv: Product2[K, V] = null// update 為偏函數val update = (hadValue: Boolean, oldValue: C) => {// 當有Value時,將oldValue與新的Value kv._2 進行merge// 若沒有Value,傳入kv._2,生成Valueif (hadValue) mergeValue(oldValue, kv._2) else createCombiner(kv._2)}while (records.hasNext) {addElementsRead()kv = records.next()// 首先使用我們的AppendOnlyMap// 在內存中對value進行聚合 map.changeValue((getPartition(kv._1), kv._1), update)// 超過閾值時寫入磁盤maybeSpillCollection(usingMap = true)}} else {// 直接把Value插入緩沖區while (records.hasNext) {addElementsRead()val kv = records.next()buffer.insert(getPartition(kv._1), kv._1, kv._2.asInstanceOf[C])maybeSpillCollection(usingMap = false)}}}

這里的createCombiner我們可以看做用kv._2生成一個Value。而mergeValue我們可以理解成為MapReduce中的combiner,即可以理解為Map端的Reduce操作,先對相同的key的Value進行聚合。

聚合算法

下面我們來深入看看聚合操作部分:

調用棧:

  • util.collection.SizeTrackingAppendOnlyMap.changeValue
    • util.collection.AppendOnlyMap.changeValue
      • util.collection.AppendOnlyMap.incrementSize
        • util.collection.AppendOnlyMap.growTable
    • util.collection.SizeTracker.afterUpdate
      • util.collection.SizeTracker.takeSample

首先是AppendOnlyMap的changeValue函數:

util.collection.SizeTrackingAppendOnlyMap.changeValue

override def changeValue(key: K, updateFunc: (Boolean, V) => V): V = {// 應用聚合算法得到newValueval newValue = super.changeValue(key, updateFunc)// 更新對 AppendOnlyMap 大小的采樣super.afterUpdate()// 返回結果newValue}

util.collection.AppendOnlyMap.changeValue

聚合算法

def changeValue(key: K, updateFunc: (Boolean, V) => V): V = {assert(!destroyed, destructionMessage)val k = key.asInstanceOf[AnyRef]if (k.eq(null)) {if (!haveNullValue) {incrementSize()}nullValue = updateFunc(haveNullValue, nullValue)haveNullValue = truereturn nullValue}// 根據k的hashCode在哈希 與 上 掩碼 得到 pos// 2*pos 為 k 應該所在的位置// 2*pos + 1 為 k 對應的 v 所在的位置var pos = rehash(k.hashCode) & maskvar i = 1while (true) {// 得到data中k所在的位置上的值curKeyval curKey = data(2 * pos)if (curKey.eq(null)) {// 若curKey為空// 得到根據 kv._2,即單個新值 生成的 newValueval newValue = updateFunc(false, null.asInstanceOf[V])data(2 * pos) = kdata(2 * pos + 1) = newValue.asInstanceOf[AnyRef]// 擴充容量incrementSize()return newValue} else if (k.eq(curKey) || k.equals(curKey)) {// 若k 與 curKey 相等// 將oldValue(data(2 * pos + 1)) 和 新的Value(kv._2) 進行聚合val newValue = updateFunc(true, data(2 * pos + 1).asInstanceOf[V])data(2 * pos + 1) = newValue.asInstanceOf[AnyRef]return newValue} else {// 若curKey 不為null,也和k不想等,// 即 hash 沖突// 則 不斷的向后遍歷 直到出現前兩種情況val delta = ipos = (pos + delta) & maski += 1}}null.asInstanceOf[V] }
  • 1

util.collection.AppendOnlyMap.incrementSize

我們再來看一下擴充容量的實現:

private def incrementSize() {curSize += 1// 當curSize大于閾值growThreshold時,// 調用growTable()if (curSize > growThreshold) {growTable()}}
  • 1

util.collection.AppendOnlyMap.growTable

protected def growTable() {生成容量翻倍的newDataval newCapacity = capacity * 2require(newCapacity <= MAXIMUM_CAPACITY, s"Can't contain more than ${growThreshold} elements")val newData = new Array[AnyRef](2 * newCapacity)// 生成newMaskval newMask = newCapacity - 1var oldPos = 0while (oldPos < capacity) {// 將舊的Data 中的數據用newMask重新計算位置,// 復制到新的Data 中 if (!data(2 * oldPos).eq(null)) {val key = data(2 * oldPos)val value = data(2 * oldPos + 1)var newPos = rehash(key.hashCode) & newMaskvar i = 1var keepGoing = truewhile (keepGoing) {val curKey = newData(2 * newPos)if (curKey.eq(null)) {newData(2 * newPos) = keynewData(2 * newPos + 1) = valuekeepGoing = false} else {val delta = inewPos = (newPos + delta) & newMaski += 1}}}oldPos += 1}// 更新data = newDatacapacity = newCapacitymask = newMaskgrowThreshold = (LOAD_FACTOR * newCapacity).toInt}
  • 1

util.collection.SizeTracker.afterUpdate

我們回過頭來看SizeTrackingAppendOnlyMap.changeValue中的更新對AppendOnlyMap大小的采樣super.afterUpdate()。所謂大小的采樣,是只一次Update后AppendOnlyMap大小的變化量。但是如果在每次如insert``update等操作后就進行計算一次AppendOnlyMap會大大降低性能。所以,這里采用了采樣估計的方法:

protected def afterUpdate(): Unit = {numUpdates += 1// 若numUpdates到達閾值,// 則進行采樣if (nextSampleNum == numUpdates) {takeSample()}}
  • 1

util.collection.SizeTracker.takeSample

private def takeSample(): Unit = {samples.enqueue(Sample(SizeEstimator.estimate(this), numUpdates))// 只用兩個采樣if (samples.size > 2) {samples.dequeue()}val bytesDelta = samples.toList.reverse match {// 估計出每次更新的變化量case latest :: previous :: tail =>(latest.size - previous.size).toDouble / (latest.numUpdates - previous.numUpdates)// 若小于 2個 樣本, 假設沒產生變化case _ => 0}// 更新bytesPerUpdate = math.max(0, bytesDelta)// 增大閾值nextSampleNum = math.ceil(numUpdates * SAMPLE_GROWTH_RATE).toLong}
  • 1
  • 10

我們再看來下估計AppendOnlyMap大小的函數:

def estimateSize(): Long = {assert(samples.nonEmpty)// 計算估計的總變化量val extrapolatedDelta = bytesPerUpdate * (numUpdates - samples.last.numUpdates)// 之前的大小 加上 估計的總變化量(samples.last.size + extrapolatedDelta).toLong}

寫緩沖區

現在我們回到insertAll,深入看看如何直接把Value插入緩沖區。

調用棧:

  • util.collection.PartitionedPairBuffer.insert
    • util.collection.PartitionedPairBuffer.growArray

util.collection.PartitionedPairBuffer.insert

def insert(partition: Int, key: K, value: V): Unit = {// 到了容量大小,調用growArray()if (curSize == capacity) {growArray()}data(2 * curSize) = (partition, key.asInstanceOf[AnyRef])data(2 * curSize + 1) = value.asInstanceOf[AnyRef]curSize += 1afterUpdate()}
  • 1

util.collection.PartitionedPairBuffer.growArray

private def growArray(): Unit = {if (capacity >= MAXIMUM_CAPACITY) {throw new IllegalStateException(s"Can't insert more than ${MAXIMUM_CAPACITY} elements")}val newCapacity =if (capacity * 2 < 0 || capacity * 2 > MAXIMUM_CAPACITY) { // OverflowMAXIMUM_CAPACITY} else {capacity * 2}// 生成翻倍容量的newArrayval newArray = new Array[AnyRef](2 * newCapacity)// 復制System.arraycopy(data, 0, newArray, 0, 2 * capacity)data = newArraycapacity = newCapacityresetSamples()}
  • 8

溢出

現在我們回到insertAll,深入看看如何將超過閾值時寫入磁盤:

調用棧:

  • util.collection.ExternalSorter.maybeSpillCollection
    • util.collection.Spillable.maybeSpill
      • util.collection.Spillable.spill
        • util.collection.ExternalSorter.spillMemoryIteratorToDisk

util.collection.ExternalSorter.maybeSpillCollection

private def maybeSpillCollection(usingMap: Boolean): Unit = {var estimatedSize = 0Lif (usingMap) {estimatedSize = map.estimateSize()if (maybeSpill(map, estimatedSize)) {map = new PartitionedAppendOnlyMap[K, C]}} else {estimatedSize = buffer.estimateSize()if (maybeSpill(buffer, estimatedSize)) {buffer = new PartitionedPairBuffer[K, C]}}if (estimatedSize > _peakMemoryUsedBytes) {_peakMemoryUsedBytes = estimatedSize}}
  • 5

util.collection.Spillable.maybeSpill

protected def maybeSpill(collection: C, currentMemory: Long): Boolean = {var shouldSpill = falseif (elementsRead % 32 == 0 && currentMemory >= myMemoryThreshold) {// 若大于閾值// amountToRequest 為要申請的內存空間val amountToRequest = 2 * currentMemory - myMemoryThresholdval granted = acquireMemory(amountToRequest)myMemoryThreshold += granted// 若果我們分配了太小的內存,// 由于 tryToAcquire 返回0// 或者 內存申請大小超過了myMemoryThreshold// 導致 依然 currentMemory >= myMemoryThreshold// 則 shouldSpillshouldSpill = currentMemory >= myMemoryThreshold}// 若元素讀取數大于閾值// 則 shouldSpillshouldSpill = shouldSpill || _elementsRead > numElementsForceSpillThresholdif (shouldSpill) {// 跟新 Spill 次數_spillCount += 1logSpillage(currentMemory)// Spill操作spill(collection)// 元素讀取數 清零_elementsRead = 0// 增加Spill的內存計數// 釋放內存_memoryBytesSpilled += currentMemoryreleaseMemory()}shouldSpill}
  • 1

util.collection.Spillable.spill

將內存中的集合spill到一個有序文件中。之后SortShuffleWriter.write中會調用sorter.writePartitionedFile來merge它們

override protected[this] def spill(collection: WritablePartitionedPairCollection[K, C]): Unit = {// 生成內存中集合的迭代器,// 這部分我們之后會深入講解val inMemoryIterator = collection.destructiveSortedWritablePartitionedIterator(comparator)// 生成spill文件,// 并將其加入數組val spillFile = spillMemoryIteratorToDisk(inMemoryIterator)spills += spillFile}
  • 1

util.collection.ExternalSorter.spillMemoryIteratorToDisk

private[this] def spillMemoryIteratorToDisk(inMemoryIterator: WritablePartitionedIterator): SpilledFile = { // 生成臨時文件 及 blockId val (blockId, file) = diskBlockManager.createTempShuffleBlock()// 這些值在每次flush后會被重置var objectsWritten: Long = 0val spillMetrics: ShuffleWriteMetrics = new ShuffleWriteMetricsval writer: DiskBlockObjectWriter =blockManager.getDiskWriter(blockId, file, serInstance, fileBufferSize, spillMetrics)// 按寫入磁盤的順序記錄分支的大小val batchSizes = new ArrayBuffer[Long]// 記錄每個分區有多少元素val elementsPerPartition = new Array[Long](numPartitions)// Flush writer 內容到磁盤,// 并更新相關變量def flush(): Unit = {val segment = writer.commitAndGet()batchSizes += segment.length_diskBytesSpilled += segment.lengthobjectsWritten = 0}var success = falsetry {// 遍歷內存集合while (inMemoryIterator.hasNext) {val partitionId = inMemoryIterator.nextPartition()require(partitionId >= 0 && partitionId < numPartitions,s"partition Id: ${partitionId} should be in the range [0, ${numPartitions})")inMemoryIterator.writeNext(writer)elementsPerPartition(partitionId) += 1objectsWritten += 1// 當寫入的元素個數 到達 批量序列化尺寸,// flushif (objectsWritten == serializerBatchSize) {flush()}}if (objectsWritten > 0) {// 遍歷結束后還有寫入// flushflush()} else {writer.revertPartialWritesAndClose()}success = true} finally {if (success) {writer.close()} else {writer.revertPartialWritesAndClose()if (file.exists()) {if (!file.delete()) {logWarning(s"Error deleting ${file}")}}}}SpilledFile(file, blockId, batchSizes.toArray, elementsPerPartition)}
  • 1

排序

我們再在回到,SortShuffleWriter.write中:

// 在外部排序中,// 有部分結果可能在內存中// 另外部分結果在一個或多個文件中// 需要將它們merge成一個大文件val partitionLengths = sorter.writePartitionedFile(blockId, tmp)

調用棧:

  • util.collection.writePartitionedFile
    • util.collection.ExternalSorter.destructiveSortedWritablePartitionedIterator
    • util.collection.ExternalSorter.partitionedIterator
      • partitionedDestructiveSortedIterator

util.collection.ExternalSorter.writePartitionedFile

我們先來深入看下writePartitionedFile,將數據加入這個ExternalSorter中,寫入一個磁盤文件:

def writePartitionedFile(blockId: BlockId,outputFile: File): Array[Long] = {// 跟蹤輸出文件的位置val lengths = new Array[Long](numPartitions)val writer = blockManager.getDiskWriter(blockId, outputFile, serInstance, fileBufferSize,context.taskMetrics().shuffleWriteMetrics)if (spills.isEmpty) {// 當只有內存中有數據時val collection = if (aggregator.isDefined) map else bufferval it = collection.destructiveSortedWritablePartitionedIterator(comparator)while (it.hasNext) {val partitionId = it.nextPartition()while (it.hasNext && it.nextPartition() == partitionId) {it.writeNext(writer)}val segment = writer.commitAndGet()lengths(partitionId) = segment.length}} else {// 否則必須進行merge-sort// 得到一個分區迭代器// 并且直接把所有數據寫入for ((id, elements) <- this.partitionedIterator) {if (elements.hasNext) {for (elem <- elements) {writer.write(elem._1, elem._2)}val segment = writer.commitAndGet()lengths(id) = segment.length}}}writer.close()context.taskMetrics().incMemoryBytesSpilled(memoryBytesSpilled)context.taskMetrics().incDiskBytesSpilled(diskBytesSpilled)context.taskMetrics().incPeakExecutionMemory(peakMemoryUsedBytes)lengths}
  • 1
  • 28

util.collection.ExternalSorter.destructiveSortedWritablePartitionedIterator

在writePartitionedFile使用destructiveSortedWritablePartitionedIterator生成了迭代器:

val it = collection.destructiveSortedWritablePartitionedIterator(comparator)

在上篇博文中提到util.collection.Spillable.spill中也使用到了它:

val inMemoryIterator = collection.destructiveSortedWritablePartitionedIterator(comparator)
  • 1

我們來看下destructiveSortedWritablePartitionedIterator:

def destructiveSortedWritablePartitionedIterator(keyComparator: Option[Comparator[K]]): WritablePartitionedIterator = {// 生成迭代器val it = partitionedDestructiveSortedIterator(keyComparator)new WritablePartitionedIterator {private[this] var cur = if (it.hasNext) it.next() else nulldef writeNext(writer: DiskBlockObjectWriter): Unit = {writer.write(cur._1._2, cur._2)cur = if (it.hasNext) it.next() else null}def hasNext(): Boolean = cur != nulldef nextPartition(): Int = cur._1._1}}
  • 1

可以看到WritablePartitionedIterator相當于partitionedDestructiveSortedIterator所返回的迭代器的代理類。destructiveSortedWritablePartitionedIterator并不返回值,而是將DiskBlockObjectWriter傳入,再進行寫。我們先把partitionedDestructiveSortedIterator放一下,往下看。

util.collection.ExternalSorter.partitionedIterator

和另外一個分支不同,這個分支是調用partitionedIterator得到分區迭代器,并且直接把所有數據寫入。我們來深入看看partitionedIterator:

def partitionedIterator: Iterator[(Int, Iterator[Product2[K, C]])] = {val usingMap = aggregator.isDefinedval collection: WritablePartitionedPairCollection[K, C] = if (usingMap) map else bufferif (spills.isEmpty) {// 當沒有spills// 按我們之前的流程 不會 加入這分支if (!ordering.isDefined) {// 若不需要對key排序// 則只對Partition進行排序groupByPartition(destructiveIterator(collection.partitionedDestructiveSortedIterator(None)))} else {// 否則需要對partition和key 進行排序groupByPartition(destructiveIterator(collection.partitionedDestructiveSortedIterator(Some(keyComparator))))}} else {// 當有spills// 需要 Merge spilled出來的那些臨時文件 和 內存中的 數據merge(spills, destructiveIterator(collection.partitionedDestructiveSortedIterator(comparator)))}}
  • 1

我們先來看下spills.isEmpty時候,兩種排序方式:

  • 只對Partition進行排序:
    partitionedDestructiveSortedIterator中傳入的是None,意思是不對key進行排序。對Partition進行排序是默認會在partitionedDestructiveSortedIterator中進行的。我們留在后面講解。
groupByPartition(destructiveIterator(collection.partitionedDestructiveSortedIterator(None)))
  • 1

Partition排序后,根據Partition的聚合:

private def groupByPartition(data: Iterator[((Int, K), C)]): Iterator[(Int, Iterator[Product2[K, C]])] ={val buffered = data.buffered(0 until numPartitions).iterator.map(p => (p, new IteratorForPartition(p, buffered)))}
  • 1
  • 2

IteratorForPartition就是對單個partion的迭代器:

private[this] class IteratorForPartition(partitionId: Int, data: BufferedIterator[((Int, K), C)])extends Iterator[Product2[K, C]]{override def hasNext: Boolean = data.hasNext && data.head._1._1 == partitionIdoverride def next(): Product2[K, C] = {if (!hasNext) {throw new NoSuchElementException}val elem = data.next()(elem._1._2, elem._2)}}
  • 對partition和key進行排序
groupByPartition(destructiveIterator(collection.partitionedDestructiveSortedIterator(Some(keyComparator))))
  • 1

partitionedDestructiveSortedIterator中傳入的是keyComparator:

private val keyComparator: Comparator[K] = ordering.getOrElse(new Comparator[K] {override def compare(a: K, b: K): Int = {val h1 = if (a == null) 0 else a.hashCode()val h2 = if (b == null) 0 else b.hashCode()if (h1 < h2) -1 else if (h1 == h2) 0 else 1}})

先根據key的hashCode進行排序,再調用groupByPartition對partition進行排序。

而對于有spills時,我們使用comparator:

private def comparator: Option[Comparator[K]] = {// 若需要排序 或者 需要 聚合if (ordering.isDefined || aggregator.isDefined) {Some(keyComparator)} else {None}}
  • 1

partitionedDestructiveSortedIterator

好了接下來我們就來看看partitionedDestructiveSortedIterator。partitionedDestructiveSortedIterator是特質WritablePartitionedPairCollection中的方法。WritablePartitionedPairCollection由PartitionedAppendOnlyMap和PartitionedPairBuffer繼承。在partitionedIterator中可以看到:

val usingMap = aggregator.isDefinedval collection: WritablePartitionedPairCollection[K, C] = if (usingMap) map else buffer
  • 1

若需要聚合,則使用PartitionedAppendOnlyMap,否則使用PartitionedPairBuffer

util.collection.PartitionedPairBuffer.partitionedDestructiveSortedIterator

我們先來看下簡單點的PartitionedPairBuffer.partitionedDestructiveSortedIterator:

override def partitionedDestructiveSortedIterator(keyComparator: Option[Comparator[K]]): Iterator[((Int, K), V)] = {val comparator = keyComparator.map(partitionKeyComparator).getOrElse(partitionComparator)// 對數據進行排序new Sorter(new KVArraySortDataFormat[(Int, K), AnyRef]).sort(data, 0, curSize, comparator)iterator}
  • 1

我們可以看到上述:

val comparator = keyComparator.map(partitionKeyComparator).getOrElse(partitionComparator)
  • 1

使用partitionKeyComparator將原來的comparator給替換了。partitionKeyComparator就是partition和key二次排序,如果傳入的keyComparator為None,那就是只對Partition進行排序:

def partitionKeyComparator[K](keyComparator: Comparator[K]): Comparator[(Int, K)] = {new Comparator[(Int, K)] {override def compare(a: (Int, K), b: (Int, K)): Int = {val partitionDiff = a._1 - b._1if (partitionDiff != 0) {partitionDiff} else {keyComparator.compare(a._2, b._2)}}}
  • 1
  • 2

之后我們使用Sort等對數據進行排序,其中用到了TimSort,在以后博文中,我們會深入講解。

最后返回迭代器iterator,其實就是簡單的按一對一對的去遍歷數據:

private def iterator(): Iterator[((Int, K), V)] = new Iterator[((Int, K), V)] {var pos = 0override def hasNext: Boolean = pos < curSizeoverride def next(): ((Int, K), V) = {if (!hasNext) {throw new NoSuchElementException}val pair = (data(2 * pos).asInstanceOf[(Int, K)], data(2 * pos + 1).asInstanceOf[V])pos += 1pair}} }
  • 1
  • 6

util.collection.PartitionedAppendOnlyMap.partitionedDestructiveSortedIterator

def partitionedDestructiveSortedIterator(keyComparator: Option[Comparator[K]]): Iterator[((Int, K), V)] = {val comparator = keyComparator.map(partitionKeyComparator).getOrElse(partitionComparator)destructiveSortedIterator(comparator)}
  • 1

util.collection.PartitionedAppendOnlyMap.destructiveSortedIterator

def destructiveSortedIterator(keyComparator: Comparator[K]): Iterator[(K, V)] = {destroyed = true// 向左整理var keyIndex, newIndex = 0while (keyIndex < capacity) {if (data(2 * keyIndex) != null) {data(2 * newIndex) = data(2 * keyIndex)data(2 * newIndex + 1) = data(2 * keyIndex + 1)newIndex += 1}keyIndex += 1}assert(curSize == newIndex + (if (haveNullValue) 1 else 0))new Sorter(new KVArraySortDataFormat[K, AnyRef]).sort(data, 0, newIndex, keyComparator)// 返回新的 Iteratornew Iterator[(K, V)] {var i = 0var nullValueReady = haveNullValuedef hasNext: Boolean = (i < newIndex || nullValueReady)def next(): (K, V) = {if (nullValueReady) {nullValueReady = false(null.asInstanceOf[K], nullValue)} else {val item = (data(2 * i).asInstanceOf[K], data(2 * i + 1).asInstanceOf[V])i += 1item}}}}

總結

以上是生活随笔為你收集整理的深入理解Spark 2.1 Core (十):Shuffle Map 端的原理与源码分析的全部內容,希望文章能夠幫你解決所遇到的問題。

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

九九免费在线观看 | 国产中文字幕免费 | 色网站在线免费观看 | 欧美亚洲精品在线观看 | 亚洲午夜精品一区二区三区电影院 | 亚洲免费国产 | 久久福利小视频 | 97精品国产97久久久久久春色 | 国产精品一区二区久久精品爱微奶 | 综合激情 | 欧美成人69av | 人人干人人搞 | 黄色av高清 | 久久成人国产精品入口 | 99精品在线免费观看 | 成人91免费视频 | 日韩在线观看一区 | 午夜视频在线观看一区二区 | www.久久精品视频 | 久久大视频 | 91网站观看| 操操操日日日干干干 | 国产精品久久久久久一区二区 | 久久在草 | 久久精品com| 国产99久久久国产精品免费看 | 日日干 天天干 | 中文字幕综合在线 | 欧美大片www | 日韩视频一区二区在线 | 8x成人免费视频 | 狠狠躁夜夜a产精品视频 | www.午夜视频| 国产精品入口a级 | 色多视频在线观看 | 日韩在线观看视频一区二区三区 | 黄色片视频在线观看 | 久久精品国产一区 | 四虎在线免费观看 | 黄色小说免费在线观看 | 91成人精品一区在线播放69 | 国产精品久久久久久久久久久免费 | 国产麻豆精品久久一二三 | 免费看国产a | 中文字幕欧美日韩va免费视频 | 超碰在线97国产 | 九九综合久久 | 午夜av大片 | 狠狠干天天色 | 亚洲乱亚洲乱妇 | 免费在线观看亚洲视频 | 97超碰资源 | 免费在线观看91 | 一级欧美黄 | 国产va精品免费观看 | 欧美成人性网 | 婷婷精品国产欧美精品亚洲人人爽 | 国产一区二区久久精品 | 中文字幕第一页在线视频 | 五月宗合网 | 狠狠色丁香婷婷综合视频 | 91精品国产亚洲 | 久久久亚洲电影 | 美女免费视频观看网站 | 国产美女久久久 | 视频在线亚洲 | 最近中文字幕完整高清 | 激情欧美一区二区三区 | 97在线观看 | 国产亚洲资源 | 国产精品av免费在线观看 | 国产免费视频一区二区裸体 | 国产精品不卡一区 | 四虎永久免费网站 | 亚洲视频专区在线 | 日本高清中文字幕有码在线 | 91免费看黄色 | 欧美 日韩 成人 | 欧美在线不卡一区 | 狠狠色噜噜狠狠狠狠 | 亚洲中字幕 | 国产毛片aaa | 久久视频一区二区 | 亚洲国产精品激情在线观看 | 日韩欧美国产免费播放 | 美女黄网久久 | 久久无码精品一区二区三区 | 国产黄色片在线免费观看 | 亚洲精品免费在线观看 | 久久99精品国产麻豆宅宅 | 国产精品九九视频 | 久草在线免费看视频 | 五月天网站在线 | 久久成人国产精品一区二区 | 国产电影黄色av | 91久久国产露脸精品国产闺蜜 | 日本一区二区三区视频在线播放 | 国产一区高清在线 | 五月天伊人网 | 久久成人毛片 | 国产毛片在线 | 欧美日韩国产一区 | 欧美a影视 | 狠狠操精品 | 免费在线观看的av网站 | 欧美最猛性xxxxx亚洲精品 | 久久久久亚洲国产精品 | 亚洲高清在线视频 | 日韩天天干 | 欧美日韩在线视频一区二区 | 久精品在线 | 亚洲成免费| 亚洲精品字幕 | 国产在线高清 | 日韩欧美电影在线观看 | 青青久视频 | 国产精品门事件 | 久久国产精品99国产精 | 97成人在线免费视频 | 日韩成人免费在线观看 | 久久久影院一区二区三区 | 日韩亚洲在线 | 久久色视频 | 激情五月伊人 | 久久这里精品视频 | 在线 日韩 av | 在线视频一区二区 | 久久精品99国产精品日本 | 免费黄在线观看 | 五月婷婷在线播放 | av大片免费看 | 日韩在线观看视频免费 | 国产精品久久久久久久久久了 | 少妇高潮流白浆在线观看 | 日韩久久电影 | 久久久91精品国产一区二区精品 | 天天操一操 | 欧美日韩视频免费 | 日韩精品视频在线观看免费 | 91人人插 | 亚洲午夜精品一区二区三区电影院 | 最新日本中文字幕 | aaa日本高清在线播放免费观看 | 日韩精品高清视频 | 日产乱码一二三区别免费 | 中文字幕在线观看完整版 | 国内三级在线观看 | 久久激情五月激情 | 狠狠操操| 免费性网站 | 亚洲视频2| 国产精品久久av | 色婷婷av国产精品 | 国产精品视频app | 久久精品理论 | 亚洲专区欧美 | 91成人免费观看视频 | 欧美大香线蕉线伊人久久 | 久草在线资源观看 | 99精品久久久久久久久久综合 | av千婊在线免费观看 | 久久99视频精品 | 日韩欧美69 | 激情五月网站 | 色婷婷综合久久久 | 狠狠的干狠狠的操 | 天天综合网 天天 | 欧美日韩伦理一区 | 久久久午夜精品理论片中文字幕 | 亚洲伦理一区 | 九九久久婷婷 | 天天色图| 国内久久久久久 | 中文字幕在线看视频 | 超碰在线国产 | 97人人爽人人 | 91一区一区三区 | 日韩在线播放av | 日本久久精品视频 | www中文在线 | 中文字幕免费在线看 | 欧美激情精品久久久久久免费印度 | 免费看污片| 久草在线最新视频 | 国产在线观看一区 | 亚在线播放中文视频 | 精品亚洲免费视频 | 五月婷婷中文网 | 国产精品自拍av | 久久综合中文色婷婷 | 超碰精品在线观看 | 伊人黄 | 日韩免费视频一区二区 | 久草网在线视频 | 国产精品手机看片 | 欧美成人h版在线观看 | 1区2区视频| 九九有精品 | 国产精品视频专区 | 国产精品九九热 | 婷婷 综合 色 | bayu135国产精品视频 | 91久久电影 | 在线免费观看视频你懂的 | 天天拍天天操 | 99久久久久久久久久 | 亚洲v欧美v国产v在线观看 | 在线免费观看视频你懂的 | 久久96国产精品久久99软件 | 日日摸日日| 欧美午夜a | 亚洲永久精品一区 | 不卡视频在线 | 亚洲成人精品影院 | 国产激情免费 | 国产精品一区二区三区99 | 国产精品久99 | 偷拍视频一区 | 一区二区在线电影 | 中文字幕在线观看网站 | 日韩av黄| 99精品视频网 | 视频直播国产精品 | 天天夜夜亚洲 | 免费黄色在线网址 | 在线国产中文 | 国产精品影音先锋 | 欧美国产日韩在线观看 | 国产又粗又长的视频 | 99精品在线视频播放 | 欧美视频国产视频 | 亚洲一区二区三区毛片 | 国产 视频 久久 | av综合网址 | 韩国av三级 | 国产在线播放一区 | 久久字幕| 午夜在线资源 | 久久久久久免费 | 国产精品自产拍在线观看蜜 | 91精品在线视频观看 | 精品久久精品久久 | 中文字幕第一页在线 | 国产美女视频免费观看的网站 | 欧美精品v国产精品 | 亚洲欧美视频在线播放 | 国精产品一二三线999 | 国产高清久久 | 欧美午夜性 | 91九色蝌蚪视频在线 | 国产一区二区视频在线播放 | 亚洲人成在线观看 | 日韩精品一区二区免费 | 欧美另类tv | 一级久久久 | 少妇性bbb搡bbb爽爽爽欧美 | 亚洲国产福利视频 | 91精品国产综合久久福利不卡 | 99精品视频免费观看 | 欧美视频在线观看免费网址 | 永久av免费在线观看 | 992tv又爽又黄的免费视频 | 亚洲激情在线 | 国产69精品久久99不卡的观看体验 | 日韩欧美精品在线 | 在线不卡中文字幕播放 | 一级免费黄视频 | 欧美a视频在线观看 | 狠狠色综合网站久久久久久久 | 国产成人一二片 | 色偷偷88888欧美精品久久 | 国产高清视频在线 | 日日日日干 | 国产一区免费观看 | 美女网站久久 | 激情久久综合 | 久久久久久久久久影视 | 久久免费高清视频 | 8090yy亚洲精品久久 | 91成人在线免费观看 | 成人性生交大片免费观看网站 | 国产在线观看免费观看 | 欧美日韩中文在线观看 | 中文字幕在线看视频国产中文版 | 91理论片午午伦夜理片久久 | 免费福利小视频 | 91色视频 | 91麻豆精品国产91久久久使用方法 | 综合网欧美 | 久久伦理视频 | 国色天香第二季 | 精品视频久久 | 国产视频久久久 | 欧美91精品国产自产 | 亚洲高清资源 | 四虎永久精品在线 | 91网页版免费观看 | 久射网 | 蜜臀久久99精品久久久久久网站 | 99高清视频有精品视频 | 国产精品一区二区精品视频免费看 | 91香蕉视频720p | 中文字幕 婷婷 | 日韩激情片在线观看 | 狠狠色丁香久久婷婷综 | 激情丁香综合 | 最新国产中文字幕 | 国产免费高清视频 | 国产综合精品一区二区三区 | 久久精品国产一区二区电影 | www国产亚洲 | 色婷婷五 | 国产精品视频永久免费播放 | 日本一区二区不卡高清 | 色吊丝av中文字幕 | 综合久久一本 | 午夜.dj高清免费观看视频 | 国产在线观看中文字幕 | 亚洲女人天堂成人av在线 | 黄色免费观看视频 | 在线精品视频免费播放 | free. 性欧美.com | 色噜噜狠狠狠狠色综合久不 | 激情五月在线视频 | 美女久久久久久 | 久久九九影视网 | 深爱婷婷激情 | 日韩在线视频观看免费 | 一区二区三区动漫 | 99久久99久久综合 | 黄色av播放 | 久久婷婷一区二区三区 | 午夜精品久久久久久久99 | 西西444www大胆无视频 | 久久你懂得 | 97成人免费视频 | 伊色综合久久之综合久久 | 毛片网站在线观看 | 国产午夜三级一区二区三 | 久久免费片 | 欧美在线一二 | 97视频在线观看网址 | 天天视频色 | 人人爽人人爱 | 伊人婷婷在线 | 91福利社区在线观看 | 在线观看日韩精品 | wwwwwww色| 日韩系列 | 九九九九九国产 | 中文字幕精品在线 | 久久电影色 | 999久久久欧美日韩黑人 | 国产香蕉视频 | 亚洲涩涩网 | 亚洲国产精品女人久久久 | 亚洲区精品视频 | 日本丶国产丶欧美色综合 | 久久综合五月婷婷 | 亚洲午夜久久久久久久久电影网 | 久久久国际精品 | 亚洲爱爱视频 | 国产精品精品视频 | 天天色天天骑天天射 | 欧美色图东方 | 97成人超碰| 成人中文字幕在线观看 | 久久99精品国产麻豆宅宅 | 日韩欧美一区二区三区在线观看 | 国产特级毛片 | 成年人电影免费看 | 国产成人av一区二区三区在线观看 | 久草在线免费看视频 | 久久成人毛片 | 精品视频久久久久久 | 日韩精品久久久久久久电影竹菊 | 亚洲精品午夜国产va久久成人 | 亚洲小视频在线观看 | 中文字幕高清免费日韩视频在线 | 色婷婷97| 97品白浆高清久久久久久 | 日韩精选在线观看 | 婷婷亚洲最大 | 欧美成人一二区 | 成人a免费 | 日韩乱码在线 | 最新精品国产 | 国产精品99免费看 | 娇妻呻吟一区二区三区 | 久久婷婷国产 | 97超碰人人在线 | 精品999在线观看 | 精品欧美在线视频 | 欧美极品少妇xbxb性爽爽视频 | 在线中文字幕网站 | 人人干人人草 | 亚洲久草网 | av一区二区三区在线观看 | 在线观看色网 | 欧美成人xxxxxxxx | a天堂免费| 久久夜视频 | 久久精品久久99精品久久 | 91大神免费在线观看 | 日韩爱爱片 | 国产日韩精品一区二区三区 | av在线播放观看 | 五月天激情在线 | 成年人网站免费在线观看 | 欧美午夜久久久 | 亚洲精品影视 | 97成人精品区在线播放 | 国产精品自产拍在线观看 | 亚洲蜜桃在线 | 国语麻豆 | 在线a亚洲视频播放在线观看 | 亚洲精品456在线播放第一页 | 天天干天天想 | 91视频 - x99av| 国产伦精品一区二区三区在线 | 91麻豆福利 | 国产视频在线观看一区 | 亚洲欧美综合 | 欧美污污网站 | 日产乱码一二三区别在线 | 欧美日韩91 | 六月激情网 | 91久久久久久久一区二区 | 一二三精品视频 | 久久久久久毛片精品免费不卡 | 欧洲精品在线视频 | 国产黄色网 | 亚洲国产成人精品在线 | 成人国产精品一区 | av解说在线观看 | 又黄又爽又色无遮挡免费 | 国产综合片 | 日韩亚洲国产中文字幕 | 黄污视频网站 | 爱爱av网| 成片免费 | av免费线看| 成人一级免费视频 | 日日干综合 | 激情av网址 | 欧美日韩免费观看一区二区三区 | 国产超碰97 | 一区二区三区日韩视频在线观看 | 国产精品视频免费在线观看 | 亚洲精品视频二区 | 中文字幕精品www乱入免费视频 | 欧美a级一区二区 | 成人精品一区二区三区电影免费 | 国产伦精品一区二区三区照片91 | 成年人视频免费在线播放 | 国产精品夜夜夜一区二区三区尤 | 成人午夜黄色影院 | 网站在线观看你们懂的 | 欧美日韩国产在线精品 | 亚洲视频电影在线 | 一区二区三区在线观看免费视频 | 西西大胆免费视频 | 国产字幕在线看 | 久久黄色影视 | 超碰资源在线 | 一级电影免费在线观看 | 在线亚洲观看 | 91污在线| 日韩黄色一区 | 日韩免费b| 一区二区三区在线观看 | av一区二区三区在线观看 | 黄色www| 日韩在线电影一区二区 | 国产精品日韩欧美一区二区 | 日韩免费视频观看 | 国产亚洲欧美在线视频 | 2018亚洲男人天堂 | 91精品国产综合久久久久久久 | 视频在线播放国产 | 亚洲干| 在线观看 国产 | 色噜噜狠狠色综合中国 | 欧洲性视频 | 欧美福利片在线观看 | 狠狠色噜噜狠狠 | 日日操操操 | 天天综合网国产 | 国内久久精品视频 | 国产无套精品久久久久久 | 色99导航 | 天天草夜夜 | 国产免费观看视频 | 麻豆精品91| 99热 精品在线 | 五月天综合在线 | 毛片网站在线看 | 久久无码av一区二区三区电影网 | 91视频网址入口 | 国产色视频一区二区三区qq号 | 久久麻豆视频 | wwwwwww黄 | 日本h视频在线观看 | 中文字幕日韩一区二区三区不卡 | 四虎海外影库www4hu | 国产97视频 | 97国产精品 | 日本精品视频在线观看 | 国产精品久久久久久久久久东京 | 国产精品免费小视频 | 中文字幕丰满人伦在线 | 久久精品一区 | 精品国产乱码久久久久久1区2匹 | 天天操夜夜操天天射 | 91激情视频在线播放 | 国产首页| 久久亚洲私人国产精品 | 国产精品密入口果冻 | 欧美日韩国产精品一区 | 亚洲精品久 | 毛片网在线播放 | 韩国精品在线 | 久热色超碰 | 国产一区二区在线播放 | 夜夜操天天摸 | 国产精品久久久久久69 | 国产精品精品久久久久久 | 免费看日韩 | 亚洲第一区精品 | www在线免费观看 | 黄色大片日本免费大片 | 麻豆国产精品永久免费视频 | 亚洲一区 影院 | 久久高清免费观看 | 国内精品久久久久影院一蜜桃 | 欧美日韩精品在线 | 免费网站黄| 日韩精品偷拍 | 色婷婷激情电影 | 国产精品久久久久久久午夜片 | 亚洲九九九在线观看 | 99r精品视频在线观看 | 免费午夜视频在线观看 | 中文字幕免费高清 | 国产日本亚洲高清 | 在线免费黄网站 | 国产成人精品999 | 日韩精品视频久久 | 久产久精国产品 | 久久久国产精品免费 | 又大又硬又黄又爽视频在线观看 | 婷婷六月在线 | 美女视频免费精品 | 日本精品一区二区三区在线播放视频 | 97网| 日韩成人中文字幕 | 免费成视频 | 日韩色视频在线观看 | 亚洲精品国久久99热 | 国内精品久久久久久久97牛牛 | 香蕉视频国产在线 | 你操综合 | 久久九九久久九九 | 久久99热精品这里久久精品 | 91福利视频久久久久 | 欧美日韩一区二区在线 | 午夜视频在线观看一区 | 国产精品黄色影片导航在线观看 | 狠狠狠狠狠狠天天爱 | 91丨九色丨国产在线观看 | 亚洲欧美日本一区二区三区 | 99精品视频中文字幕 | 亚洲一二视频 | ,久久福利影视 | 精品免费国产一区二区三区四区 | 亚洲日本va中文字幕 | 欧美激情精品久久久久 | 玖玖国产精品视频 | 在线观看免费色 | 亚洲高清在线精品 | 久久久久亚洲精品中文字幕 | 欧美日韩一级视频 | 99精品99 | 五月婷婷综合在线 | 成人黄色片免费 | 99热国内精品 | 美女免费视频一区二区 | 在线网站黄 | 久久久久伊人 | 99久久精品国产一区二区成人 | 日本mv大片欧洲mv大片 | 国产免费久久久久 | 精品国产乱码久久久久久1区二区 | 亚洲国产无 | 日韩一区二区三区在线看 | 日韩欧美一区二区三区视频 | 九九视频免费 | 观看免费av| 免费在线观看黄网站 | 日本在线观看一区 | 在线国产能看的 | 国产成视频在线观看 | 国产精品午夜久久 | 在线观看免费福利 | 五月天综合网站 | 亚洲黄色在线 | 91在线播放视频 | 中文字幕电影高清在线观看 | 一级片视频免费观看 | 国产亚洲精品成人av久久ww | 美女福利视频一区二区 | 午夜精品久久久久久久久久 | 久久伊99综合婷婷久久伊 | 激情动态 | 国产一区二区在线免费视频 | 中文字幕专区高清在线观看 | 成人三级视频 | 国产精品v欧美精品 | 中文字幕av最新 | 国产精品免费久久久 | 国产精品九九久久久久久久 | 天天天天干 | 亚洲激情小视频 | 久久久久久久久久久福利 | 国产手机视频在线观看 | 国产韩国精品一区二区三区 | 成人av一区二区在线观看 | 亚洲成人黄色在线观看 | 韩国精品在线观看 | 91av久久 | 精品电影一区 | 精品一区二区三区久久 | 国产中文字幕网 | 国产黄色免费观看 | 超碰97人人干 | 午夜影院一级片 | 国际精品久久久久 | 免费黄在线看 | 国产成人精品一二三区 | 欧美日韩激情视频8区 | 一区二区三区免费在线播放 | 在线亚洲午夜片av大片 | 国产精品久久嫩一区二区免费 | 亚洲精品九九 | 国产小视频免费观看 | 日韩精品在线免费播放 | 综合久久久| 天天插一插 | 亚洲 中文 在线 精品 | 四虎永久免费网站 | 久久爱导航 | 一区二区三区免费在线播放 | 这里只有精品视频在线观看 | 日韩精品一区二区三区在线视频 | www.com久久久 | 天天草综合 | 丁香六月中文字幕 | 日韩免费中文字幕 | 日韩在线视频不卡 | 一级a毛片高清视频 | 日韩高清无线码2023 | 国产精品扒开做爽爽的视频 | 久久久国产精品网站 | 国产精品九九九九九九 | 99久久精品免费看国产一区二区三区 | 黄色电影网站在线观看 | 欧美精品免费视频 | 国产精品videossex国产高清 | 国产一区二区久久精品 | 亚洲天堂网站视频 | 日韩欧美一级二级 | 亚洲一级电影视频 | 久久av一区二区三区亚洲 | 国产亚洲精品bv在线观看 | 中文一区二区三区在线观看 | 国内精品久久久久影院优 | 成人av电影在线 | 久久久视频在线 | 日躁夜躁狠狠躁2001 | 日韩av电影手机在线观看 | 99超碰在线播放 | 国产精品自产拍在线观看 | 五月婷婷激情六月 | 日韩高清精品一区二区 | 日韩国产精品久久久久久亚洲 | 久久久久久美女 | 五月天九九 | 亚洲国产成人精品在线 | 国产欧美精品在线观看 | 五月天六月婷婷 | 日本在线成人 | 成片免费观看视频 | 日本激情视频中文字幕 | 成人黄色在线视频 | 中文字幕日韩无 | 亚洲精品小区久久久久久 | 亚洲日日日 | 福利久久久| 欧美一级专区免费大片 | 欧美a视频在线观看 | 免费特级黄色片 | 制服丝袜欧美 | 玖玖色在线观看 | 久久99热精品这里久久精品 | 国产欧美三级 | 99精品欧美一区二区 | 久久精品亚洲 | 91视频免费看网站 | 久久久久久高潮国产精品视 | 欧美一级乱黄 | 伊人影院得得 | 国产一区二区在线免费播放 | 99久久综合精品五月天 | 99精品国产99久久久久久福利 | 摸bbb搡bbb搡bbbb| 成人一区二区三区中文字幕 | 天天综合人人 | 国产精品不卡在线 | 91在线免费看片 | 欧美日韩国产免费视频 | 国产中文字幕三区 | 5月丁香婷婷综合 | 国产精品21区 | 亚洲精品国产片 | 九九热在线观看视频 | 国产日韩三级 | 黄色免费网站大全 | 91污污 | 亚洲成人精品av | 五月天激情在线 | www.伊人色.com | 日韩免费观看高清 | 欧美日韩一级在线 | www.天天干 | 欧美在线视频不卡 | 激情电影影院 | 国产999精品| 黄a在线观看 | 久久一区国产 | 西西444www大胆高清视频 | 亚洲精品美女久久久久 | 91久久在线观看 | 精品国产一区二区三区久久 | 精品你懂的 | 999精品| 日韩在线观 | 中文字幕 国产专区 | 人人干干人人 | 成人一区二区在线观看 | 久久在线免费观看视频 | 黄色免费av | 国产剧情一区二区 | 欧美精品999 | 日韩免费电影一区二区 | a在线一区 | 亚洲国产三级在线 | 午夜久久精品 | 国产精品入口麻豆www | 欧美日韩在线免费观看视频 | 久操伊人 | 久久99精品久久久久蜜臀 | 视频精品一区二区三区 | 这里只有精彩视频 | 天天操偷偷干 | 国产精品视频地址 | 久久99久久99精品免观看软件 | 国产精品一区二区中文字幕 | 亚洲春色综合另类校园电影 | 五月婷婷综合激情 | 99精品视频精品精品视频 | 九九九九色 | 天天爽天天摸 | 狠狠操狠狠干天天操 | 国产精品网在线观看 | 国产成人精品一区二三区 | 中文av影院 | 欧美激情xxxx | 一区二区日韩av | 成人一级免费电影 | 久久人人97超碰精品888 | 日韩在线观看精品 | 国产精品美女999 | 在线视频麻豆 | 激情五月在线观看 | 中文字幕三区 | 日本xxxxav| 国产一区二区在线免费观看 | 天天av资源| 精品日韩av| 午夜精品一区二区三区在线观看 | 国产破处视频在线播放 | 国产在线 一区二区三区 | 99亚洲国产 | 日本精品久久久久影院 | 色综合网在线 | 91麻豆视频| 久久男人视频 | 久久久久久精 | 久久婷婷国产 | 日韩中文字幕免费视频 | 夜添久久精品亚洲国产精品 | 狠狠操欧美 | 97视频在线播放 | 欧美日韩亚洲在线观看 | 激情综合色综合久久综合 | 久久人视频 | 久久乱码卡一卡2卡三卡四 五月婷婷久 | 中文字幕在线高清 | 四虎欧美| 亚洲狠狠操| 成人免费观看大片 | 成 人 黄 色 视频 免费观看 | 亚洲精品黄网站 | 91av蜜桃| 亚洲国产精品成人va在线观看 | 亚洲精品电影在线 | 成人国产精品av | 国产精品嫩草影院99网站 | 色偷偷88888欧美精品久久久 | 91亚洲精品在线 | 欧美污在线观看 | 久久看视频 | 一级黄网 | av片一区二区 | 爱爱av网| 国产精品综合久久 | 在线视频一二三 | 久草成人在线 | 亚洲午夜电影网 | 欧美日韩一区三区 | 欧美aaa大片 | 99久久99| 欧美日韩免费视频 | 久久午夜网 | 欧美激情视频在线免费观看 | 在线免费三级 | 99婷婷狠狠成为人免费视频 | 天天色图| 国精产品一二三线999 | 狠狠色噜噜狠狠狠狠2021天天 | 久久久网页 | 国产精品一级在线 | 精品黄色在线观看 | 成人免费网视频 | 黄色日批网站 | 蜜桃视频成人在线观看 | 久久精品视频在线免费观看 | av片一区二区 | 亚洲视频电影在线 | 午夜视频在线观看一区二区 | 国产美腿白丝袜足在线av | 欧美婷婷综合 | 亚洲美女视频在线观看 | 国产一区麻豆 | 日韩免费成人av | 狠狠躁日日躁狂躁夜夜躁av | 亚洲欧美精品在线 | 亚洲aⅴ乱码精品成人区 | 国产亚洲aⅴaaaaaa毛片 | 天天射天天干天天操 | 亚洲日本成人网 | 国产欧美精品一区二区三区四区 | 成人在线播放免费观看 | 国产精品久久久久久久久久久久久 | 中文字幕在线久一本久 | 97超碰成人在线 | 伊人久久国产精品 | 在线看一区二区 | 亚洲精品玖玖玖av在线看 | 免费在线成人av | 婷五月激情 | 天天爽夜夜操 | 欧美精品久久久久久久久久白贞 | 97人人模人人爽人人少妇 | 91精品一区二区三区蜜桃 | 在线成人一区二区 | 色噜噜在线观看视频 | 91九色porn在线资源 | 日本乱码在线 | 亚洲三级毛片 | 日韩午夜电影院 | 久久久免费观看 | 精品国内自产拍在线观看视频 | 欧美成人黄色 | 国产69精品久久99的直播节目 | 国产精品一区在线播放 | 久久国产一区 | 一区二区三区久久精品 | 日日干av| 国产xxxx做受性欧美88 | 欧美另类色图 | 揉bbb玩bbb少妇bbb | 国产亚洲亚洲 | 一区二区三区电影 | 69国产盗摄一区二区三区五区 | 黄a在线看 | 中文字幕电影网 | 一区国产精品 | 丁香六月色 | 在线免费av网站 | 久久综合久久综合这里只有精品 | 99精品在线免费观看 | 久久免费视频在线观看30 | 国产亚洲成人网 | 欧美日韩中文在线观看 | 成人午夜电影免费在线观看 | 亚洲涩综合 | 国产在线不卡一区 | 中文字幕在线高清 | 中文在线www | 亚洲色图22p | 日韩一区二区三区在线看 | 国产中文字幕av | 高清av免费看 | 国产精品久久久久毛片大屁完整版 | 91精品在线免费 | 在线观看视频免费大全 | 最近最新最好看中文视频 | 日韩有码在线观看视频 | 91av视频导航 | 天堂av在线免费观看 | 日韩视频一区二区在线 | 69精品在线观看 | 一本一本久久a久久 | 亚洲激情影院 | 中文字幕在线观看免费高清完整版 | 99国产在线观看 | 欧美aaa视频| 亚洲视频999 | 在线色资源| 精品无人国产偷自产在线 | 国产亚洲精品久久 | 美女视频黄在线 | 精品一区二区日韩 | 日韩极品视频在线观看 | 国产精品国产三级国产aⅴ9色 | 天天操天天添 | 久久8| h动漫中文字幕 | 77国产精品| 亚洲成人一二三 | 天天操天天曰 | 999久久久久久久久6666 | 丰满少妇在线观看 | 国产在线不卡精品 | 99c视频高清免费观看 | 99久久久久国产精品免费 | 成人资源在线观看 | 日韩精品在线看 | 国产精品99爱 | 成人精品一区二区三区中文字幕 | 91九色国产| 成av人电影 | va视频在线 | 久久久久久久久久毛片 | 亚洲精品国偷自产在线99热 | 美女网站色 | 中文av网| aa级黄色大片| 久草免费在线观看视频 | 国产成视频在线观看 | 国产精品免费在线观看视频 | 国产亚洲成人网 | 精品自拍sae8—视频 | 久久久免费精品 | 91精品视频一区二区三区 | 丁香5月婷婷 | 国产成人一区二 | 天天爽天天射 | 日韩黄色大片在线观看 | 黄色成人91 | 中文字幕日韩精品有码视频 | 色在线中文字幕 | 日韩黄视频 | 午夜av色 | 天天色天天操综合 | 99免费在线播放99久久免费 | 国产成人精品av在线观 | 精品一区二区三区香蕉蜜桃 | 免费成人在线电影 | 丁香在线视频 | 99精品视频精品精品视频 | 美女精品在线观看 | 亚洲精品看片 | 一区二区欧美激情 | 四虎国产精品免费观看视频优播 | av免费在线观 | 日韩v欧美v日本v亚洲v国产v | 中文字幕第一页在线 | aa一级片| 国产一区欧美二区 | 国内精品视频在线 | 国产高清无av久久 | 97偷拍视频| 香蕉影视| 综合久久精品 | 成人国产综合 | 91视频 - x99av|