国产无遮挡裸体免费直播视频,久久精品国产蜜臀av,动漫在线视频一区二区,欧亚日韩一区二区三区,久艹在线 免费视频,国产精品美女网站免费,正在播放 97超级视频在线观看,斗破苍穹年番在线观看免费,51最新乱码中文字幕

Lucene源碼系列多值編碼壓縮算法實(shí)例詳解

 更新時(shí)間:2022年11月16日 09:57:15   作者:滄叔解碼  
這篇文章主要為大家介紹了Lucene源碼系列多值編碼壓縮算法實(shí)例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

背景

在Lucene中,涉及到索引文件生成的時(shí)候,會(huì)看到比較多的PackedInts.Encoder,PackedWriter,DirectWriter,DirectMonotonicWriter等等對(duì)多個(gè)long進(jìn)行壓縮編碼解碼的使用,但是它們之間有什么區(qū)別和聯(lián)系呢?本文就是詳細(xì)介紹說(shuō)明Lucene中對(duì)正整數(shù)(int或者long)數(shù)組的壓縮編碼解碼方式。

雖然限制了只對(duì)正整數(shù)有用,但是其他數(shù)值可以通過(guò)一些轉(zhuǎn)化,先轉(zhuǎn)成正整數(shù),然后再使用本文介紹的壓縮編碼。比如,負(fù)整數(shù)可以通過(guò)zigzag先做一次編碼,float或者double可以通過(guò)把二進(jìn)制按int或者long來(lái)解析進(jìn)行預(yù)處理。

特別說(shuō)明

本文中說(shuō)到的壓縮,壓縮編碼器,編碼器指的都是同一個(gè)東西。另外,本文相關(guān)的源碼主要都是在進(jìn)行一些位運(yùn)算,我會(huì)有注釋,但是如果實(shí)在看不太懂的話,可以自己舉例子走一遍源碼就清楚了。

本文的重點(diǎn)還是壓縮編碼的思想,以及幾個(gè)工具類的不同使用場(chǎng)景的理解。

前置知識(shí)

單值壓縮中,我們已經(jīng)說(shuō)過(guò)了,壓縮其實(shí)就是去除無(wú)用的信息,而對(duì)于數(shù)值來(lái)說(shuō),高位的連續(xù)0其實(shí)就是無(wú)用的信息。

假設(shè)有l(wèi)ong數(shù)組中的值為:

	long[] values = {10, 4, 9, 16, 132};

它們的十進(jìn)制和二進(jìn)制分別是:

十進(jìn)制二進(jìn)制
1000000000 00000000 00000000 00000000 00000000 00000000 00000000 00001010
400000000 00000000 00000000 00000000 00000000 00000000 00000000 00000100
900000000 00000000 00000000 00000000 00000000 00000000 00000000 00001001
1600000000 00000000 00000000 00000000 00000000 00000000 00000000 00010000
58000000000 00000000 00000000 00000000 00000000 00000000 00000010 01000100

如果按照正常的二進(jìn)制編碼存儲(chǔ),一個(gè)long需要64bits,則values數(shù)組需要5*64=320bits。但是我們觀察所有value的二進(jìn)制,發(fā)現(xiàn)最大的580也只有最后10bits是有效的,因此,我們可以用10bits表示每一個(gè)value,這樣一共只需要50bits就能完成對(duì)values數(shù)組的存儲(chǔ),如下表所示:

十進(jìn)制壓縮編碼
1000 00001010
400 00000100
900 00001001
1600 00010000
58010 10000100

所以,所謂的壓縮編碼其實(shí)就是只保留value的有效位。接下來(lái)先介紹兩個(gè)很重要的概念:

  • bitsPerValue:要壓縮編碼的整數(shù)數(shù)組中,最大的值的二進(jìn)制有效位數(shù),如上面例子中的就是10。
  • block:把用來(lái)存儲(chǔ)壓縮編碼后的value的空間分成一個(gè)個(gè)block,block大小一般是long或者byte。

Lucene中根據(jù)一個(gè)value編碼壓縮之后的存儲(chǔ)是否跨越多個(gè)block,分為兩類壓縮編解碼。下面我們以上面的例子來(lái)介紹這兩種編解碼的區(qū)別。

假設(shè)我們block的大小是int,也就是一個(gè)block是32位。在上面的例子中,每個(gè)value需要10bits存儲(chǔ)空間,所以一個(gè)int中能完整存儲(chǔ)3個(gè)value之后,還剩下2bits,兩個(gè)編碼的區(qū)別就是是否要使用這2bits,具體結(jié)果如下圖所示:

從上面的例子中,我們總結(jié)下兩種編碼的特點(diǎn):

單個(gè)value跨block(BulkOperationPacked和BulkOperationPacked*,*是從1-24。)

  • 除了最后一個(gè)block,沒(méi)有浪費(fèi)存儲(chǔ)空間
  • 增加解碼的復(fù)雜度

實(shí)現(xiàn)類:

bitsPerValue>24bitsPerValue>24bitsPerValue>24時(shí),使用BulkOperationPacked

bitsPerValue∈[1,24]bitsPerValue\in[1,24]bitsPerValue∈[1,24]時(shí),使用BulkOperationPacked*,*是從1到24。

單個(gè)value不跨block

  • 存在空間浪費(fèi),浪費(fèi)程度和bitsPerValue有關(guān)
  • 解碼比較簡(jiǎn)單
  • 實(shí)現(xiàn)類:BulkOperationPackedSingleBlock

這兩種不同的編碼方式在Lucene中用兩個(gè)枚舉類來(lái)區(qū)分,PACKED和PACKED_SINGLE_BLOCK。

public enum Format {
  // 所有的值都是連續(xù)存儲(chǔ)在一起的,允許一個(gè)value編碼結(jié)果跨越多個(gè)block
  PACKED(0) {
	// 忽略不相關(guān)的代碼
  },
  @Deprecated  
  // 一個(gè)value一個(gè)block  
  PACKED_SINGLE_BLOCK(1) {
    // 忽略不相關(guān)的代碼
  };
  // 忽略不相關(guān)的代碼  
}

特別需要注意的是,在9.0.0版本之前,Lucene中是嚴(yán)格實(shí)現(xiàn)了這兩種策略,但是從9.0.0開(kāi)始,Lucene廢棄了PACKED_SINGLE_BLOCK,因此在本文中,我們重點(diǎn)介紹PACKED。

總覽

Lucene中對(duì)批量的正整數(shù)壓縮編碼設(shè)置了幾種可選的壓縮模式,有不同的內(nèi)存浪費(fèi)和不同的編解碼速度,實(shí)際上就是用空間換時(shí)間程度的差異:

  • FASTEST:這種編碼速度允許浪費(fèi)7倍的內(nèi)存,是編碼速度最快的一種模式,在實(shí)現(xiàn)上基本就是一個(gè)block一個(gè)value。
  • FAST:這種編碼速度允許浪費(fèi)50%的內(nèi)存,會(huì)選擇一種比較合適的bitsPerValue來(lái)獲取速度較快的編碼器。
  • DEFAULT:這種編碼速度允許浪費(fèi)25%的內(nèi)存,編碼速度中規(guī)中矩。
  • COMPACT:這種編碼不允許浪費(fèi)任何內(nèi)存,編碼速度是最慢的。

在PackedInts中有個(gè)工具方法fastestFormatAndBits,接受要編碼的數(shù)據(jù)量,bitsPerValue,以及預(yù)期可浪費(fèi)的最大內(nèi)存來(lái)調(diào)整bitsPerValue,從而獲得當(dāng)前條件下最快的編碼器。

// acceptableOverheadRatio:允許編碼結(jié)果浪費(fèi)的內(nèi)存比率
public static FormatAndBits fastestFormatAndBits(
    int valueCount, int bitsPerValue, float acceptableOverheadRatio) {
  if (valueCount == -1) { // 如果不清楚要編碼的數(shù)據(jù)量,就把valueCount設(shè)為-1,會(huì)按最大的數(shù)量來(lái)估計(jì)
    valueCount = Integer.MAX_VALUE;
  }
  // acceptableOverheadRatio不能小于COMPACT
  acceptableOverheadRatio = Math.max(COMPACT, acceptableOverheadRatio);
  // acceptableOverheadRatio不能超過(guò)FASTEST
  acceptableOverheadRatio = Math.min(FASTEST, acceptableOverheadRatio);
  // 每個(gè)value可以允許浪費(fèi)的比例  
  float acceptableOverheadPerValue = acceptableOverheadRatio * bitsPerValue; 
  // 每個(gè)value最大可能使用的bit數(shù)  
  int maxBitsPerValue = bitsPerValue + (int) acceptableOverheadPerValue;
  int actualBitsPerValue = -1;
  // block一般是int或者是long,所以8,16,32,64的編碼速度是比較快的
  if (bitsPerValue <= 8 && maxBitsPerValue >= 8) {
    actualBitsPerValue = 8;
  } else if (bitsPerValue <= 16 && maxBitsPerValue >= 16) {
    actualBitsPerValue = 16;
  } else if (bitsPerValue <= 32 && maxBitsPerValue >= 32) {
    actualBitsPerValue = 32;
  } else if (bitsPerValue <= 64 && maxBitsPerValue >= 64) {
    actualBitsPerValue = 64;
  } else {
    actualBitsPerValue = bitsPerValue;
  }
  // 強(qiáng)制使用Format.PACKED,已經(jīng)不使用PACKED_SINGLE_BLOCK了
  return new FormatAndBits(Format.PACKED, actualBitsPerValue);
}

有了format和bitsPerValue之后,BulkOperation中使用策略模式實(shí)現(xiàn)了獲取相應(yīng)編碼器的邏輯:

public static Encoder getEncoder(Format format, int version, int bitsPerValue) {
  checkVersion(version);
  return BulkOperation.of(format, bitsPerValue);
}

可以看到策略模式中就是獲取bitsPerValue對(duì)應(yīng)的提前創(chuàng)建好的編解碼器:

public static BulkOperation of(PackedInts.Format format, int bitsPerValue) {
  switch (format) {
    case PACKED:
      assert packedBulkOps[bitsPerValue - 1] != null;
      return packedBulkOps[bitsPerValue - 1];
    case PACKED_SINGLE_BLOCK: // 棄用,不用看了
      assert packedSingleBlockBulkOps[bitsPerValue - 1] != null;
      return packedSingleBlockBulkOps[bitsPerValue - 1];
    default:
      throw new AssertionError();
  }
}

對(duì)于bitsPerValue的每個(gè)可能值,都有一個(gè)對(duì)應(yīng)的編碼器:

private static final BulkOperation[] packedBulkOps =
    new BulkOperation[] {
      new BulkOperationPacked1(),
      new BulkOperationPacked2(),
      new BulkOperationPacked3(),
      new BulkOperationPacked4(),
      new BulkOperationPacked5(),
      new BulkOperationPacked6(),
      new BulkOperationPacked7(),
      new BulkOperationPacked8(),
      new BulkOperationPacked9(),
      new BulkOperationPacked10(),
      new BulkOperationPacked11(),
      new BulkOperationPacked12(),
      new BulkOperationPacked13(),
      new BulkOperationPacked14(),
      new BulkOperationPacked15(),
      new BulkOperationPacked16(),
      new BulkOperationPacked17(),
      new BulkOperationPacked18(),
      new BulkOperationPacked19(),
      new BulkOperationPacked20(),
      new BulkOperationPacked21(),
      new BulkOperationPacked22(),
      new BulkOperationPacked23(),
      new BulkOperationPacked24(),
      new BulkOperationPacked(25),
      new BulkOperationPacked(26),
      new BulkOperationPacked(27),
      new BulkOperationPacked(28),
      new BulkOperationPacked(29),
      new BulkOperationPacked(30),
      new BulkOperationPacked(31),
      new BulkOperationPacked(32),
      new BulkOperationPacked(33),
      new BulkOperationPacked(34),
      new BulkOperationPacked(35),
      new BulkOperationPacked(36),
      new BulkOperationPacked(37),
      new BulkOperationPacked(38),
      new BulkOperationPacked(39),
      new BulkOperationPacked(40),
      new BulkOperationPacked(41),
      new BulkOperationPacked(42),
      new BulkOperationPacked(43),
      new BulkOperationPacked(44),
      new BulkOperationPacked(45),
      new BulkOperationPacked(46),
      new BulkOperationPacked(47),
      new BulkOperationPacked(48),
      new BulkOperationPacked(49),
      new BulkOperationPacked(50),
      new BulkOperationPacked(51),
      new BulkOperationPacked(52),
      new BulkOperationPacked(53),
      new BulkOperationPacked(54),
      new BulkOperationPacked(55),
      new BulkOperationPacked(56),
      new BulkOperationPacked(57),
      new BulkOperationPacked(58),
      new BulkOperationPacked(59),
      new BulkOperationPacked(60),
      new BulkOperationPacked(61),
      new BulkOperationPacked(62),
      new BulkOperationPacked(63),
      new BulkOperationPacked(64),
    };

前面我們已經(jīng)說(shuō)過(guò)了,PACKED_SINGLE_BLOCK被棄用了,因此我們下面重點(diǎn)介紹PACKED的情況。

編解碼

上面的BulkOperation中預(yù)置的編解碼器看著很復(fù)雜,其實(shí)是比較簡(jiǎn)單的。從名字中帶BulkOperation就知道,這是對(duì)數(shù)據(jù)的批量操作,也就是對(duì)數(shù)據(jù)的批量編解碼,下面我們來(lái)詳細(xì)介紹相關(guān)的類的邏輯。

BulkOperation

在BulkOperation中除了預(yù)置所有bitsPerValue對(duì)應(yīng)的編解碼器和根據(jù)bitsPerValue獲取對(duì)應(yīng)的編解碼器之外,還有一個(gè)比較重要的方法,涉及到一個(gè)概念:iteration。

在理解iteration之前,我們先想想,如果要編碼的數(shù)據(jù)量特別大,我們一次性全加載進(jìn)內(nèi)存,豈不是會(huì)占用很大的內(nèi)存空間,甚至可能出現(xiàn)OOM。因此,需要控制編碼過(guò)程中內(nèi)存的使用情況,這就涉及到了iteration。

各個(gè)編碼器在自己內(nèi)部會(huì)限定每個(gè)iteration最多能夠處理的value個(gè)數(shù),通過(guò)下面的方法可以算出在一定的內(nèi)存限制條件下最多可以使用多少個(gè)iteration,這樣就可以間接知道一次編碼可以處理多少個(gè)value了。

// ramBudget限制在編碼過(guò)程中使用的內(nèi)存上限,單位是字節(jié)
public final int computeIterations(int valueCount, int ramBudget) {
  // byteBlockCount():一個(gè)iteration需要幾個(gè)byte存儲(chǔ)編碼后的結(jié)果
  // byteValueCount(): 一個(gè)iteration需要幾個(gè)byte存儲(chǔ)待編碼的value個(gè)數(shù),value是都按long類型處理
  // 這里先算出來(lái)ramBudget可以支撐的最大的iteration,并沒(méi)有考慮實(shí)際的valueCount,后面再調(diào)整
  final int iterations = ramBudget / (byteBlockCount() + 8 * byteValueCount());
  if (iterations == 0) { // 至少是1
    return 1;
  } else if ((iterations - 1) * byteValueCount() >= valueCount) { // 如果不需要那么多的迭代次數(shù)
    // 根據(jù)真實(shí)的valueCount計(jì)算合適的iteration  
    return (int) Math.ceil((double) valueCount / byteValueCount()); 
  } else {
    return iterations;
  }
}

BulkOperationPacked

BulkOperationPacked是一個(gè)通用的編解碼器實(shí)現(xiàn),它支持所有的bitsPerValue。

成員變量

需要重點(diǎn)理解下longBlockCount,longValueCount,byteBlockCount,byteValueCount四個(gè)變量的意思,具體這四個(gè)變量是得來(lái)的,邏輯在構(gòu)造器中。

private final int bitsPerValue;
// 當(dāng)block是long的時(shí)候,一個(gè)iteration最少會(huì)使用幾個(gè)block
private final int longBlockCount;
// 一個(gè)iteration中,使用longBlockCount個(gè)block可以編碼的value數(shù)量
private final int longValueCount;
// 當(dāng)block是byte的時(shí)候,一個(gè)iteration最少會(huì)使用幾個(gè)block
private final int byteBlockCount;
// 一個(gè)iteration中,使用byteBlockCount個(gè)block可以編碼的value數(shù)量
private final int byteValueCount;
// 用來(lái)獲取value中有效位的掩碼
private final long mask;
// mask的int格式
private final int intMask;

構(gòu)造器

在構(gòu)造器中,根據(jù)固定的規(guī)則計(jì)算longBlockCount,longValueCount,byteBlockCount,byteValueCount。

這里我不知道為什么使用這樣的規(guī)則,如果有朋友知道其中奧秘的可以評(píng)論分享。

  public BulkOperationPacked(int bitsPerValue) {
    this.bitsPerValue = bitsPerValue;
    assert bitsPerValue > 0 && bitsPerValue <= 64;
    int blocks = bitsPerValue;
    while ((blocks & 1) == 0) { // 如果是2的倍數(shù)則除以2,直到不是2的倍數(shù)
      blocks >>>= 1;
    }
    // 一次迭代用幾個(gè)long  
    this.longBlockCount = blocks;
    // 一次迭代可以處理幾個(gè)value  
    this.longValueCount = 64 * longBlockCount / bitsPerValue;
    // 一次迭代用幾個(gè)byte  
    int byteBlockCount = 8 * longBlockCount;
    // 一次迭代用處理幾個(gè)value  
    int byteValueCount = longValueCount;
    while ((byteBlockCount & 1) == 0 && (byteValueCount & 1) == 0) { // 縮小成直到不是2的倍數(shù)
      byteBlockCount >>>= 1;
      byteValueCount >>>= 1;
    }
    this.byteBlockCount = byteBlockCount;
    this.byteValueCount = byteValueCount;
    // 算掩碼  
    if (bitsPerValue == 64) {
      this.mask = ~0L;
    } else {
      this.mask = (1L << bitsPerValue) - 1;
    }
    this.intMask = (int) mask;
    assert longValueCount * bitsPerValue == 64 * longBlockCount;
  }

編碼

根據(jù)block大小和要處理的value的類型的不同,一共有四個(gè)編碼方法:

以long為block的單位,編碼long數(shù)組

  public void encode(
      long[] values, int valuesOffset, long[] blocks, int blocksOffset, int iterations) {
    // 初始化下一個(gè)block的值  
    long nextBlock = 0;
    // 當(dāng)前block中還剩下多少位可以用來(lái)編碼  
    int bitsLeft = 64;
    // longValueCount * iterations:總共可以處理的數(shù)據(jù)  
    for (int i = 0; i < longValueCount * iterations; ++i) {
      // 當(dāng)前block使用bitsPerValue來(lái)進(jìn)行編碼  
      bitsLeft -= bitsPerValue;
      if (bitsLeft > 0) { // 當(dāng)前block放下當(dāng)前value后還有空間
        // 把當(dāng)前value放到當(dāng)前block中  
        nextBlock |= values[valuesOffset++] << bitsLeft;
      } else if (bitsLeft == 0) { // 剛好可以存放最后一個(gè)數(shù)據(jù)的編碼結(jié)果
        nextBlock |= values[valuesOffset++]; // 最后一個(gè)數(shù)據(jù)的編碼結(jié)果放在nextBlock中
        blocks[blocksOffset++] = nextBlock; // 存儲(chǔ)block
        nextBlock = 0; // 重置block
        bitsLeft = 64; // 重置bitsLeft
      } else { // 需要跨block處理
        nextBlock |= values[valuesOffset] >>> -bitsLeft; // 當(dāng)前block剩下的空間存儲(chǔ)部分value
        blocks[blocksOffset++] = nextBlock; 
        // (1L << -bitsLeft) - 1):取出value剩余數(shù)據(jù)的掩碼
        // (values[valuesOffset++] & ((1L << -bitsLeft) - 1)): 取出剩余數(shù)據(jù)
        // << (64 + bitsLeft):放到最前面
        nextBlock = (values[valuesOffset++] & ((1L << -bitsLeft) - 1)) << (64 + bitsLeft);
        // 更新block可用空間  
        bitsLeft += 64;
      }
    }
  }

以long為block的單位,編碼int數(shù)組

和上面的方法邏輯其實(shí)一模一樣,只是在處理value的時(shí)候,先和0xFFFFFFFFL做按位與運(yùn)算轉(zhuǎn)成long。

public void encode(
    int[] values, int valuesOffset, long[] blocks, int blocksOffset, int iterations) {
  long nextBlock = 0;
  int bitsLeft = 64;
  for (int i = 0; i < longValueCount * iterations; ++i) {
    bitsLeft -= bitsPerValue;
    if (bitsLeft > 0) {
      nextBlock |= (values[valuesOffset++] & 0xFFFFFFFFL) << bitsLeft;
    } else if (bitsLeft == 0) {
      nextBlock |= (values[valuesOffset++] & 0xFFFFFFFFL);
      blocks[blocksOffset++] = nextBlock;
      nextBlock = 0;
      bitsLeft = 64;
    } else { // bitsLeft < 0
      nextBlock |= (values[valuesOffset] & 0xFFFFFFFFL) >>> -bitsLeft;
      blocks[blocksOffset++] = nextBlock;
      nextBlock = (values[valuesOffset++] & ((1L << -bitsLeft) - 1)) << (64 + bitsLeft);
      bitsLeft += 64;
    }
  }
}

以byte為block的單位,編碼long數(shù)組

public void encode(
    long[] values, int valuesOffset, byte[] blocks, int blocksOffset, int iterations) {
  int nextBlock = 0;
  int bitsLeft = 8;
  for (int i = 0; i < byteValueCount * iterations; ++i) {
    final long v = values[valuesOffset++];
    assert PackedInts.unsignedBitsRequired(v) <= bitsPerValue;
    if (bitsPerValue < bitsLeft) { // 如果當(dāng)前block還可以容納value的編碼
      nextBlock |= v << (bitsLeft - bitsPerValue);
      bitsLeft -= bitsPerValue;
    } else {
      // 除了當(dāng)前block可以容納之外,還剩下多少bits
      int bits = bitsPerValue - bitsLeft;
      // 當(dāng)前block剩下的空間把value的高位部分先存  
      blocks[blocksOffset++] = (byte) (nextBlock | (v >>> bits));
      while (bits >= 8) { // 如果還需要處理的bits超過(guò)一個(gè)byte
        bits -= 8;
        blocks[blocksOffset++] = (byte) (v >>> bits);
      }
      // 剩下的bits不足一個(gè)byte
      bitsLeft = 8 - bits;
      //  (v & ((1L << bits) - 1):取出當(dāng)前value的高bits位
      // 存放在nextBlock的高bits位  
      nextBlock = (int) ((v & ((1L << bits) - 1)) << bitsLeft);
    }
  }
  assert bitsLeft == 8;
}

以byte為block的單位,編碼int數(shù)組

和上面的方法邏輯一模一樣

public void encode(
    int[] values, int valuesOffset, byte[] blocks, int blocksOffset, int iterations) {
  int nextBlock = 0;
  int bitsLeft = 8;
  for (int i = 0; i < byteValueCount * iterations; ++i) {
    final int v = values[valuesOffset++];
    assert PackedInts.bitsRequired(v & 0xFFFFFFFFL) <= bitsPerValue;
    if (bitsPerValue < bitsLeft) {
      nextBlock |= v << (bitsLeft - bitsPerValue);
      bitsLeft -= bitsPerValue;
    } else {
      int bits = bitsPerValue - bitsLeft;
      blocks[blocksOffset++] = (byte) (nextBlock | (v >>> bits));
      while (bits >= 8) {
        bits -= 8;
        blocks[blocksOffset++] = (byte) (v >>> bits);
      }
      bitsLeft = 8 - bits;
      nextBlock = (v & ((1 << bits) - 1)) << bitsLeft;
    }
  }
  assert bitsLeft == 8;
}

解碼

同樣地,和編碼一一對(duì)應(yīng),也有四個(gè)解碼方法:

以long為block的單位,解碼成long數(shù)組

public void decode(
    long[] blocks, int blocksOffset, long[] values, int valuesOffset, int iterations) {
  // 當(dāng)前block中還剩多少位未解碼  
  int bitsLeft = 64;
  for (int i = 0; i < longValueCount * iterations; ++i) {
    bitsLeft -= bitsPerValue;
    if (bitsLeft < 0) { // 跨block的解碼
      values[valuesOffset++] =
          ((blocks[blocksOffset++] & ((1L << (bitsPerValue + bitsLeft)) - 1)) << -bitsLeft)
              | (blocks[blocksOffset] >>> (64 + bitsLeft));
      bitsLeft += 64;
    } else {
      values[valuesOffset++] = (blocks[blocksOffset] >>> bitsLeft) & mask;
    }
  }
}

以long為block的單位,解碼成int數(shù)組

邏輯和上面的方法一樣。

public void decode(
    long[] blocks, int blocksOffset, int[] values, int valuesOffset, int iterations) {
  if (bitsPerValue > 32) {
    throw new UnsupportedOperationException(
        "Cannot decode " + bitsPerValue + "-bits values into an int[]");
  }
  int bitsLeft = 64;
  for (int i = 0; i < longValueCount * iterations; ++i) {
    bitsLeft -= bitsPerValue;
    if (bitsLeft < 0) {
      values[valuesOffset++] =
          (int)
              (((blocks[blocksOffset++] & ((1L << (bitsPerValue + bitsLeft)) - 1)) << -bitsLeft)
                  | (blocks[blocksOffset] >>> (64 + bitsLeft)));
      bitsLeft += 64;
    } else {
      values[valuesOffset++] = (int) ((blocks[blocksOffset] >>> bitsLeft) & mask);
    }
  }
}

以byte為block的單位,解碼成long數(shù)組

public void decode(
    byte[] blocks, int blocksOffset, long[] values, int valuesOffset, int iterations) {
  long nextValue = 0L;
  int bitsLeft = bitsPerValue;
  for (int i = 0; i < iterations * byteBlockCount; ++i) {
    final long bytes = blocks[blocksOffset++] & 0xFFL;
    if (bitsLeft > 8) {
      bitsLeft -= 8;
      nextValue |= bytes << bitsLeft;
    } else {
      int bits = 8 - bitsLeft;
      values[valuesOffset++] = nextValue | (bytes >>> bits);
      while (bits >= bitsPerValue) {
        bits -= bitsPerValue;
        values[valuesOffset++] = (bytes >>> bits) & mask;
      }
      bitsLeft = bitsPerValue - bits;
      nextValue = (bytes & ((1L << bits) - 1)) << bitsLeft;
    }
  }
  assert bitsLeft == bitsPerValue;
}

以byte為block的單位,解碼成int數(shù)組

public void decode(
    byte[] blocks, int blocksOffset, int[] values, int valuesOffset, int iterations) {
  int nextValue = 0;
  int bitsLeft = bitsPerValue;
  for (int i = 0; i < iterations * byteBlockCount; ++i) {
    final int bytes = blocks[blocksOffset++] & 0xFF;
    if (bitsLeft > 8) {
      bitsLeft -= 8;
      nextValue |= bytes << bitsLeft;
    } else {
      int bits = 8 - bitsLeft;
      values[valuesOffset++] = nextValue | (bytes >>> bits);
      while (bits >= bitsPerValue) {
        bits -= bitsPerValue;
        values[valuesOffset++] = (bytes >>> bits) & intMask;
      }
      bitsLeft = bitsPerValue - bits;
      nextValue = (bytes & ((1 << bits) - 1)) << bitsLeft;
    }
  }
  assert bitsLeft == bitsPerValue;
}

BulkOperationPacked*

BulkOperationPacked中的實(shí)現(xiàn)支持所有bitsPerValue的值,BulkOperationPacked*繼承了BulkOperationPacked,對(duì)于解碼的實(shí)現(xiàn)針對(duì)特定的bitsPerValue有了特定的實(shí)現(xiàn),性能比通用實(shí)現(xiàn)更高效(猜測(cè)是特殊實(shí)現(xiàn)沒(méi)有分支,避免了分支預(yù)測(cè)錯(cuò)誤的性能損耗),那為什么只對(duì)解碼做針對(duì)實(shí)現(xiàn),編碼怎么不一起實(shí)現(xiàn)?編碼也是可以做針對(duì)實(shí)現(xiàn)的,但是編碼只在構(gòu)建索引的時(shí)候使用一次,而解碼是會(huì)多次使用的,因此只對(duì)解碼做了特殊實(shí)現(xiàn)。那為什么只對(duì)bitsPerValue是25以下的做特殊實(shí)現(xiàn),這個(gè)不清楚,25以上不常見(jiàn)?

雖然Lucene中實(shí)現(xiàn)了24個(gè)BulkOperationPacked*,但是所有BulkOperationPacked*的主要邏輯都相同:一個(gè)iteration可以處理幾個(gè)value產(chǎn)生幾個(gè)block是已知的(在BulkOperationPacked的構(gòu)造器中計(jì)算得到),然后直接從這些block解碼得到所有的value,下面我們看一個(gè)例子就好:

當(dāng)bitsPerValue=2的時(shí)候,我們可以知道:

longBlockCount=1

longValueCount=32

byteBlockCount=1

byteValueCount=4

所以當(dāng)以long為block的單位時(shí),一個(gè)iteration可以從1個(gè)block中解析出32個(gè)value,當(dāng)以byte為block的單位是,可以從1個(gè)block中解析出4個(gè)value。

final class BulkOperationPacked2 extends BulkOperationPacked {
  public BulkOperationPacked2() {
    super(2);
  }
  @Override
  public void decode(
      long[] blocks, int blocksOffset, int[] values, int valuesOffset, int iterations) {
    for (int i = 0; i < iterations; ++i) {
      // 只會(huì)用到一個(gè)block  
      final long block = blocks[blocksOffset++];
      // 需要解析出32個(gè)value   
      for (int shift = 62; shift >= 0; shift -= 2) {
        values[valuesOffset++] = (int) ((block >>> shift) & 3);
      }
    }
  }
  @Override
  public void decode(
      byte[] blocks, int blocksOffset, int[] values, int valuesOffset, int iterations) {
    for (int j = 0; j < iterations; ++j) {
      // 只會(huì)用到一個(gè)block  
      final byte block = blocks[blocksOffset++];
      // 需要解析出4個(gè)value  
      values[valuesOffset++] = (block >>> 6) & 3;
      values[valuesOffset++] = (block >>> 4) & 3;
      values[valuesOffset++] = (block >>> 2) & 3;
      values[valuesOffset++] = block & 3;
    }
  }
  @Override
  public void decode(
      long[] blocks, int blocksOffset, long[] values, int valuesOffset, int iterations) {
    for (int i = 0; i < iterations; ++i) {
      // 只會(huì)用到一個(gè)block  
      final long block = blocks[blocksOffset++];
      // 需要解析出32個(gè)value   
      for (int shift = 62; shift >= 0; shift -= 2) {
        values[valuesOffset++] = (block >>> shift) & 3;
      }
    }
  }
  @Override
  public void decode(
      byte[] blocks, int blocksOffset, long[] values, int valuesOffset, int iterations) {
    for (int j = 0; j < iterations; ++j) {
      // 只會(huì)用到一個(gè)block 
      final byte block = blocks[blocksOffset++];
      // 需要解析出4個(gè)value  
      values[valuesOffset++] = (block >>> 6) & 3;
      values[valuesOffset++] = (block >>> 4) & 3;
      values[valuesOffset++] = (block >>> 2) & 3;
      values[valuesOffset++] = block & 3;
    }
  }
}

應(yīng)用

有了上面批量編解碼的基礎(chǔ)邏輯,我們?cè)倏纯椿诰幗獯a器有哪些應(yīng)用,而這些應(yīng)用之間有什么區(qū)別,各自又是針對(duì)什么場(chǎng)景的?

PackedWriter

PackedWriter是已知所要處理的數(shù)據(jù)的bitsPerValue,根據(jù)這個(gè)bitsPerValue獲取對(duì)應(yīng)的編碼器,因此所有的數(shù)據(jù)都使用一樣的編碼器。

PackedWriter適用于所有數(shù)據(jù)量大小差不多的情況,如果數(shù)據(jù)中存在少量值比較大的,則會(huì)影響壓縮效果,因?yàn)樗衯alue是按最大的值來(lái)計(jì)算bitsPerValue的。

final class PackedWriter extends PackedInts.Writer {
  // 是否構(gòu)建結(jié)束
  boolean finished;
  // 9.0.0之后的版本都是Packed
  final PackedInts.Format format;
  // 根據(jù)format和bitsPerValue決定encoder  
  final BulkOperation encoder;
  // 編碼結(jié)果的buffer  
  final byte[] nextBlocks;
  // 待編碼的value的buffer  
  final long[] nextValues;
  // 通過(guò)編碼器計(jì)算得到的iterations  
  final int iterations;
  // nextValues可以寫(xiě)入的位置 
  int off;
  // 已處理的value的個(gè)數(shù)  
  int written;
  PackedWriter(
      PackedInts.Format format, DataOutput out, int valueCount, int bitsPerValue, int mem) {
    super(out, valueCount, bitsPerValue);
    this.format = format;
    encoder = BulkOperation.of(format, bitsPerValue);
    iterations = encoder.computeIterations(valueCount, mem);
    // 創(chuàng)建合適的大小
    nextBlocks = new byte[iterations * encoder.byteBlockCount()];
    nextValues = new long[iterations * encoder.byteValueCount()];
    off = 0;
    written = 0;
    finished = false;
  }
  @Override
  protected PackedInts.Format getFormat() {
    return format;
  }
  @Override
  public void add(long v) throws IOException {
    if (valueCount != -1 && written >= valueCount) {
      throw new EOFException("Writing past end of stream");
    }
    // 把value臨時(shí)存儲(chǔ)到nextValues中,等到編碼  
    nextValues[off++] = v;
    if (off == nextValues.length) { // 如果nextValues滿了,則進(jìn)行編碼
      flush();
    }
    ++written;
  }
  @Override
  public void finish() throws IOException {
    if (valueCount != -1) { // 如果數(shù)據(jù)沒(méi)有達(dá)到預(yù)期的數(shù)量,則補(bǔ)0
      while (written < valueCount) {
        add(0L);
      }
    }
    flush();
    finished = true;
  }
  private void flush() throws IOException {
    // 編碼  
    encoder.encode(nextValues, 0, nextBlocks, 0, iterations);
    // 以byte作為block的單位,一共幾個(gè)byte  
    final int blockCount = (int) format.byteCount(PackedInts.VERSION_CURRENT, off, bitsPerValue);
    // 編碼的結(jié)果持久化  
    out.writeBytes(nextBlocks, blockCount);
    // 清空nextValues  
    Arrays.fill(nextValues, 0L);
    off = 0;
  }
  @Override
  public int ord() {
    return written - 1;
  }
}

分段處理

需要說(shuō)明下按照類名原本應(yīng)該是分block的,但是在我們底層編解碼中涉及到block的概念,如果這里再來(lái)一個(gè)block,則可能會(huì)混淆,所以在這里我們可以理解是把要處理的數(shù)據(jù)流分成多段處理。PackedWriter其實(shí)也 可以理解是分段的(設(shè)定內(nèi)存閾值下可以處理的數(shù)據(jù)量),但是它每段都是使用一樣的編碼器,而這個(gè)小節(jié)要介紹的,每一段是可以使用不一樣的編碼器。

AbstractBlockPackedWriter

前面我們介紹PackedWriter的時(shí)候說(shuō),如果大批量數(shù)據(jù)中存在少量的數(shù)值較大的value會(huì)影響壓縮效果。針對(duì)這種情況的處理,Lucene按分段處理,每段包含部分value,如果存在少量的數(shù)值較大,則只會(huì)影響其所在的段中,不會(huì)影響所有的數(shù)據(jù)的編碼。具體我們看下源碼,注意下面的源碼中忽略一些無(wú)關(guān)緊要的代碼:

abstract class AbstractBlockPackedWriter {
  // 用來(lái)持久化編碼結(jié)果的
  protected DataOutput out;
  // 暫存待編碼的value  
  protected final long[] values;
  // 存儲(chǔ)編碼后的block  
  protected byte[] blocks;
  protected int off;
  // 處理的第幾個(gè)value  
  protected long ord;
  protected boolean finished;
  // blockSize:段大小  
  protected AbstractBlockPackedWriter(DataOutput out, int blockSize) {
    checkBlockSize(blockSize, MIN_BLOCK_SIZE, MAX_BLOCK_SIZE);
    reset(out);
    values = new long[blockSize];
  }
  public void add(long l) throws IOException {
    checkNotFinished();
    if (off == values.length) { // 如果values滿了,也就是已經(jīng)收集到了一段數(shù)據(jù)了,則進(jìn)行編碼
      flush();
    }
    values[off++] = l;
    ++ord;
  }
  public void finish() throws IOException {
    checkNotFinished();
    if (off > 0) { // 如果有數(shù)據(jù)則flush
      flush();
    }
    finished = true;
  }
  // flush中計(jì)算的就是當(dāng)前段的bitsPerValue,由子類實(shí)現(xiàn)  
  protected abstract void flush() throws IOException;
  // 執(zhí)行編碼,在flush中被調(diào)用,bitsRequired是flush中計(jì)算得到的
  protected final void writeValues(int bitsRequired) throws IOException {
    // 每一段都有自己的編碼器 
    final PackedInts.Encoder encoder =
        PackedInts.getEncoder(PackedInts.Format.PACKED, PackedInts.VERSION_CURRENT, bitsRequired);
    final int iterations = values.length / encoder.byteValueCount();
    final int blockSize = encoder.byteBlockCount() * iterations;
    if (blocks == null || blocks.length < blockSize) {
      blocks = new byte[blockSize];
    }
    if (off < values.length) {
      Arrays.fill(values, off, values.length, 0L);
    }
    encoder.encode(values, 0, blocks, 0, iterations);
    final int blockCount =
        (int) PackedInts.Format.PACKED.byteCount(PackedInts.VERSION_CURRENT, off, bitsRequired);
    // 持久化編碼后的數(shù)據(jù)  
    out.writeBytes(blocks, blockCount);
  }
}

AbstractBlockPackedWriter有兩個(gè)實(shí)現(xiàn)類,分別是通用實(shí)現(xiàn)和處理單調(diào)遞增數(shù)據(jù)的,其實(shí)就是針對(duì)不同的場(chǎng)景計(jì)算bitsPerValue的邏輯不一樣。

BlockPackedWriter

BlockPackedWriter是通用的分段實(shí)現(xiàn),它先找出最小值,所有待編碼的value都減去最小值,這樣就可以縮小bitsPerValue,達(dá)到更優(yōu)的編碼效果。

public final class BlockPackedWriter extends AbstractBlockPackedWriter {
  public BlockPackedWriter(DataOutput out, int blockSize) {
    super(out, blockSize);
  }
  @Override
  protected void flush() throws IOException {
    long min = Long.MAX_VALUE, max = Long.MIN_VALUE;
    for (int i = 0; i < off; ++i) { // 尋找最大值和最小值
      min = Math.min(values[i], min);
      max = Math.max(values[i], max);
    }
    // 最大值和最小值的差值  
    final long delta = max - min;
    // 差值的bitsRequired,下面會(huì)把所有的value更新為value和min的差值,所以bitsRequired就是更新后的bitsPerValue
    int bitsRequired = delta == 0 ? 0 : PackedInts.unsignedBitsRequired(delta);
    if (bitsRequired == 64) { // 如果最大和最小的差值都需要64bits來(lái)存儲(chǔ),那就不用去更新value了
      min = 0L;
    } else if (min > 0L) {
      // 讓min盡量小,因?yàn)閙in會(huì)用VLong編碼,這樣盡可能使用更少的字節(jié)  
      min = Math.max(0L, max - PackedInts.maxValue(bitsRequired));
    }
    // bitsRequired左移一位,最后一位是標(biāo)記位,標(biāo)記min是否是0
    final int token = (bitsRequired << BPV_SHIFT) | (min == 0 ? MIN_VALUE_EQUALS_0 : 0);
    out.writeByte((byte) token);
    if (min != 0) { // VLong編碼min
      writeVLong(out, zigZagEncode(min) - 1);
    }
    if (bitsRequired > 0) {
      if (min != 0) {
        for (int i = 0; i < off; ++i) { // 所有的value更新為與最小值的差值
          values[i] -= min;
        }
      }
      // 編碼并持久化  
      writeValues(bitsRequired);
    }
    off = 0;
  }
}

MonotonicBlockPackedWriter

MonotonicBlockPackedWriter處理的是數(shù)據(jù)是遞增的情況,MonotonicBlockPackedWriter先算每?jī)蓚€(gè)value之間的平均步長(zhǎng),通過(guò)這個(gè)平均步長(zhǎng)就可以計(jì)算每個(gè)位置的期望expectValue,然后把value都更新為真實(shí)value和期望expectValue的差值,這樣可能會(huì)有負(fù)數(shù)的value,所以需要一步是找出理論上的最小值min,保證所有的value和expectValue差值都是正數(shù)。

public final class MonotonicBlockPackedWriter extends AbstractBlockPackedWriter {
  public MonotonicBlockPackedWriter(DataOutput out, int blockSize) {
    super(out, blockSize);
  }
  @Override
  protected void flush() throws IOException {
    // 計(jì)算平均步長(zhǎng)
    final float avg = off == 1 ? 0f : (float) (values[off - 1] - values[0]) / (off - 1);
    long min = values[0];
    for (int i = 1; i < off; ++i) { // 計(jì)算出期望的最小值,保證根據(jù)平均步長(zhǎng)計(jì)算每個(gè)位置的期望值都是正數(shù)
      final long actual = values[i];
      // 相同位置的期望value  
      final long expected = expected(min, avg, i);
      if (expected > actual) { // 如果期望值大于實(shí)際值
        // 調(diào)小期望的最小值  
        min -= (expected - actual);
      }
    }
    // 最大差值,用來(lái)計(jì)算bitsPerValue的
    long maxDelta = 0;
    for (int i = 0; i < off; ++i) { // 所有的值都更新為和期望值的差值
      values[i] = values[i] - expected(min, avg, i);
      maxDelta = Math.max(maxDelta, values[i]);
    }
    out.writeZLong(min);
    out.writeInt(Float.floatToIntBits(avg));
    if (maxDelta == 0) { // 所有的值都一樣
      out.writeVInt(0);
    } else {
      final int bitsRequired = PackedInts.bitsRequired(maxDelta);
      out.writeVInt(bitsRequired);
      // 編碼并持久化  
      writeValues(bitsRequired);
    }
    off = 0;
  }
}

DirectWriter

DirectWriter比較特殊,它沒(méi)有使用BulkOperationPacked系列的編碼器。DirectWriter綜合考慮空間浪費(fèi)和編碼解碼的速度,并沒(méi)有支持全部的bitsPerValue,它支持的bitsPerValue的列表是:1, 2, 4, 8, 12, 16, 20, 24, 28, 32, 40, 48, 56, 64。

在存儲(chǔ)的時(shí)候,按照bitsPerValue不同的值分為3種處理邏輯,每種處理邏輯的block都是不一樣的:

bitsPerValue是8的倍數(shù)時(shí): 8, 16, 24, 32, 40, 48, 56, 64

  • 40,48,56,64的時(shí)候,用long來(lái)存儲(chǔ)一個(gè)value,此時(shí)block就是long,并且一個(gè)block中只有一個(gè)value。
  • 24,32的時(shí)候,用int來(lái)存儲(chǔ)一個(gè)value,此時(shí)block就是int,并且一個(gè)block中只有一個(gè)value。
  • 16的時(shí)候,用short來(lái)存儲(chǔ)一個(gè)value,此時(shí)block就是short,并且一個(gè)block中只有一個(gè)value。
  • 8的時(shí)候,用byte來(lái)存儲(chǔ)一個(gè)value,此時(shí)block就是byte,并且一個(gè)block中只有一個(gè)value。

bitsPerValue是1,2,4

用long來(lái)存儲(chǔ)多個(gè)value,此時(shí)block就是long,一個(gè)block中有64/bitsPerValue個(gè)value。

bitsPerValue是12,20,28

  • 12:用int存儲(chǔ)兩個(gè)value,此時(shí)block就是int,一個(gè)block中有兩個(gè)value。
  • 20,28:用long存儲(chǔ)兩個(gè)value,此時(shí)block就是long,一個(gè)block中有兩個(gè)value。

那如果是其他bitsPerValue的值,DirectWriter會(huì)找上述支持的列表中大于bitsPerValue的第一個(gè)來(lái)進(jìn)行處理,無(wú)非是壓縮率低一些。比如當(dāng)前的bitsPerValue=17,則DirectWriter會(huì)使用20的方案來(lái)處理。

public final class DirectWriter {
  // 每個(gè)value使用幾個(gè)bit存儲(chǔ)  
  final int bitsPerValue;
  // 有幾個(gè)需要存儲(chǔ)的value  
  final long numValues;
  // 數(shù)據(jù)存儲(chǔ)的目的地  
  final DataOutput output;
  // 統(tǒng)計(jì)已經(jīng)處理了幾個(gè)數(shù)據(jù)
  long count;
  // 是否已經(jīng)處理完成了
  boolean finished;
  // nextValues的下一個(gè)寫(xiě)入的位置
  int off;
  // 存儲(chǔ)block  
  final byte[] nextBlocks;
  // 臨時(shí)存儲(chǔ)數(shù)據(jù),待構(gòu)建成block  
  final long[] nextValues;
  // DirectWriter支持的bitsPerValue參數(shù)列表  
  static final int[] SUPPORTED_BITS_PER_VALUE =
      new int[] {1, 2, 4, 8, 12, 16, 20, 24, 28, 32, 40, 48, 56, 64};
  DirectWriter(DataOutput output, long numValues, int bitsPerValue) {
    this.output = output;
    this.numValues = numValues;
    this.bitsPerValue = bitsPerValue;
    // 可用的內(nèi)存限制,是nextBlocks和nextValues兩個(gè)數(shù)組合起來(lái)的大小限制
    final int memoryBudgetInBits = Math.multiplyExact(Byte.SIZE, PackedInts.DEFAULT_BUFFER_SIZE);
    // nextValues的大小
    int bufferSize = memoryBudgetInBits / (Long.SIZE + bitsPerValue);
    // 把bufferSize調(diào)整成下一個(gè)64的倍數(shù)的值
    bufferSize = Math.toIntExact(bufferSize + 63) & 0xFFFFFFC0;
    nextValues = new long[bufferSize];
    // add 7 bytes in the end so that any value could be written as a long
    nextBlocks = new byte[bufferSize * bitsPerValue / Byte.SIZE + Long.BYTES - 1];
  }
  // 新增一個(gè)value
  public void add(long l) throws IOException {
    // 校驗(yàn)是否超出了一開(kāi)始設(shè)定的value總數(shù)
    if (count >= numValues) {
      throw new EOFException("Writing past end of stream");
    }
    // 把數(shù)據(jù)暫存在nextValues中  
    nextValues[off++] = l;
    // 如果nextValues滿了,則進(jìn)行flush編碼  
    if (off == nextValues.length) {
      flush();
    }
    // 處理的value數(shù)+1
    count++;
  }
  // 構(gòu)建一個(gè)block  
  private void flush() throws IOException {
    if (off == 0) {
      return;
    }
    // 把off之后的所有位置設(shè)為0
    Arrays.fill(nextValues, off, nextValues.length, 0L);
    // 將nextValues中的數(shù)據(jù)編碼存儲(chǔ)到nextBlocks中  
    encode(nextValues, off, nextBlocks, bitsPerValue);
    // 一共需要幾個(gè)byte存儲(chǔ)所有編碼好的數(shù)據(jù)  
    final int blockCount =
        (int) PackedInts.Format.PACKED.byteCount(PackedInts.VERSION_CURRENT, off, bitsPerValue);
    // 存儲(chǔ)編碼后的數(shù)據(jù)  
    output.writeBytes(nextBlocks, blockCount);
    off = 0;
  }
  private static void encode(long[] nextValues, int upTo, byte[] nextBlocks, int bitsPerValue) {
    if ((bitsPerValue & 7) == 0) {// bitsPerValue的值是8的倍數(shù): 8, 16, 24, 32, 40, 48, 56, 64
      // 一個(gè)value需要幾個(gè)byte  
      final int bytesPerValue = bitsPerValue / Byte.SIZE;
      // i是nextValues中的下標(biāo)  
      // o是在nextBlocks中下一個(gè)寫(xiě)入的下標(biāo)  
      for (int i = 0, o = 0; i < upTo; ++i, o += bytesPerValue) {
        final long l = nextValues[i];
        if (bitsPerValue > Integer.SIZE) { // 大于32位的都用long存
          BitUtil.VH_LE_LONG.set(nextBlocks, o, l);
        } else if (bitsPerValue > Short.SIZE) { // 24,32都用int存
          BitUtil.VH_LE_INT.set(nextBlocks, o, (int) l);
        } else if (bitsPerValue > Byte.SIZE) { // 16用short存
          BitUtil.VH_LE_SHORT.set(nextBlocks, o, (short) l);
        } else { // 8用byte存
          nextBlocks[o] = (byte) l;
        }
      }
    } else if (bitsPerValue < 8) {// bitsPerValue的值是 1, 2, 4
      // 一個(gè)long可以存幾個(gè)value  
      final int valuesPerLong = Long.SIZE / bitsPerValue;
      // o是在nextBlocks中的下一個(gè)可以寫(xiě)入的位置  
      for (int i = 0, o = 0; i < upTo; i += valuesPerLong, o += Long.BYTES) {
        long v = 0;
        // 構(gòu)建一個(gè)block,一個(gè)block就是一個(gè)long  
        for (int j = 0; j < valuesPerLong; ++j) {
          v |= nextValues[i + j] << (bitsPerValue * j);
        }
        BitUtil.VH_LE_LONG.set(nextBlocks, o, v);
      }
    } else {// bitsPerValue的值是 12, 20, 28,此時(shí)不是一個(gè)字節(jié)的整數(shù)倍,所以是2個(gè)2個(gè)處理
      // 2個(gè)value需要幾個(gè)字節(jié)  
      final int numBytesFor2Values = bitsPerValue * 2 / Byte.SIZE;
      // o是在nextBlocks中下一個(gè)可以寫(xiě)入的位置  
      for (int i = 0, o = 0; i < upTo; i += 2, o += numBytesFor2Values) {
        final long l1 = nextValues[i];
        final long l2 = nextValues[i + 1];
        // 2個(gè)合成一個(gè)  
        final long merged = l1 | (l2 << bitsPerValue);
        if (bitsPerValue <= Integer.SIZE / 2) { // 如果bitsPerValue=12,則以int的方式存兩個(gè)value
          BitUtil.VH_LE_INT.set(nextBlocks, o, (int) merged);
        } else { // bitsPerValue=20或者28,則以long的方式存兩個(gè)value
          BitUtil.VH_LE_LONG.set(nextBlocks, o, merged);
        }
      }
    }
  }
  public void finish() throws IOException {
    if (count != numValues) {
      throw new IllegalStateException(
          "Wrong number of values added, expected: " + numValues + ", got: " + count);
    }
    // 為nextValues中剩下的value編碼  
    flush();
    // DirectWriter的編碼方案存在填充字節(jié)
    int paddingBitsNeeded;
    if (bitsPerValue > Integer.SIZE) {
      paddingBitsNeeded = Long.SIZE - bitsPerValue;
    } else if (bitsPerValue > Short.SIZE) {
      paddingBitsNeeded = Integer.SIZE - bitsPerValue;
    } else if (bitsPerValue > Byte.SIZE) {
      paddingBitsNeeded = Short.SIZE - bitsPerValue;
    } else {
      paddingBitsNeeded = 0;
    }
    final int paddingBytesNeeded = (paddingBitsNeeded + Byte.SIZE - 1) / Byte.SIZE;
    for (int i = 0; i < paddingBytesNeeded; i++) {
      output.writeByte((byte) 0);
    }
    finished = true;
  }
  // 獲取一個(gè)DirectWriter的實(shí)例
  public static DirectWriter getInstance(DataOutput output, long numValues, int bitsPerValue) {
    if (Arrays.binarySearch(SUPPORTED_BITS_PER_VALUE, bitsPerValue) < 0) {
      throw new IllegalArgumentException(
          "Unsupported bitsPerValue " + bitsPerValue + ". Did you use bitsRequired?");
    }
    return new DirectWriter(output, numValues, bitsPerValue);
  }
  // 如果bitsRequired是不支持的,則需要糾正
  private static int roundBits(int bitsRequired) {
    int index = Arrays.binarySearch(SUPPORTED_BITS_PER_VALUE, bitsRequired);
    if (index < 0) { // 如果bitsRequired不是支持的參數(shù),則使用最接近且大于bitsRequired的參數(shù)
      return SUPPORTED_BITS_PER_VALUE[-index - 1];
    } else {
      return bitsRequired; // 如果是支持的,則使用它自己
    }
  }
  // 通過(guò)有符號(hào)的最大值計(jì)算需要的bitsPerValue
  public static int bitsRequired(long maxValue) {
    return roundBits(PackedInts.bitsRequired(maxValue));
  }
  // 通過(guò)無(wú)符號(hào)的最大值計(jì)算需要的bitsPerValue
  public static int unsignedBitsRequired(long maxValue) {
    return roundBits(PackedInts.unsignedBitsRequired(maxValue));
  }
}

DirectMonotonicWriter

DirectMonotonicWriter對(duì)單調(diào)遞增的數(shù)據(jù)有預(yù)處理,邏輯和MonotonicBlockPackedWriter非常相似,編碼使用的是DirectWriter。

public final class DirectMonotonicWriter {
  public static final int MIN_BLOCK_SHIFT = 2;
  public static final int MAX_BLOCK_SHIFT = 22;
  final IndexOutput meta;
  final IndexOutput data;
  final long numValues;
  final long baseDataPointer;
  final long[] buffer;
  int bufferSize;
  long count;
  boolean finished;
  long previous = Long.MIN_VALUE;
  DirectMonotonicWriter(IndexOutput metaOut, IndexOutput dataOut, long numValues, int blockShift) {
    if (blockShift < MIN_BLOCK_SHIFT || blockShift > MAX_BLOCK_SHIFT) {
      throw new IllegalArgumentException(
          "blockShift must be in ["
              + MIN_BLOCK_SHIFT
              + "-"
              + MAX_BLOCK_SHIFT
              + "], got "
              + blockShift);
    }
    if (numValues < 0) {
      throw new IllegalArgumentException("numValues can't be negative, got " + numValues);
    }
    final long numBlocks = numValues == 0 ? 0 : ((numValues - 1) >>> blockShift) + 1;
    if (numBlocks > ArrayUtil.MAX_ARRAY_LENGTH) {
      throw new IllegalArgumentException(
          "blockShift is too low for the provided number of values: blockShift="
              + blockShift
              + ", numValues="
              + numValues
              + ", MAX_ARRAY_LENGTH="
              + ArrayUtil.MAX_ARRAY_LENGTH);
    }
    this.meta = metaOut;
    this.data = dataOut;
    this.numValues = numValues;
    final int blockSize = 1 << blockShift;
    this.buffer = new long[(int) Math.min(numValues, blockSize)];
    this.bufferSize = 0;
    this.baseDataPointer = dataOut.getFilePointer();
  }
  private void flush() throws IOException {
    assert bufferSize != 0;
    // 估計(jì)數(shù)值遞增的平均步長(zhǎng)
    final float avgInc =
        (float) ((double) (buffer[bufferSize - 1] - buffer[0]) / Math.max(1, bufferSize - 1));
    for (int i = 0; i < bufferSize; ++i) {
      // 按平均步長(zhǎng)算的當(dāng)前位置的期望值  
      final long expected = (long) (avgInc * (long) i);
      // 當(dāng)前真實(shí)值和期望值之差,注意此時(shí)buffer中可能存在負(fù)數(shù) 
      buffer[i] -= expected;
    }
    // 尋找最小的值  
    long min = buffer[0];
    for (int i = 1; i < bufferSize; ++i) {
      min = Math.min(buffer[i], min);
    }
    long maxDelta = 0;
    for (int i = 0; i < bufferSize; ++i) {
      buffer[i] -= min; // 經(jīng)過(guò)這一步,buffer中所有的都是正數(shù)
      maxDelta |= buffer[i];
    }
    meta.writeLong(min);
    meta.writeInt(Float.floatToIntBits(avgInc));
    meta.writeLong(data.getFilePointer() - baseDataPointer);
    if (maxDelta == 0) { // 經(jīng)過(guò)預(yù)處理之后,所有的值都相等
      meta.writeByte((byte) 0);
    } else { // 使用DirectWriter來(lái)進(jìn)行編碼
      final int bitsRequired = DirectWriter.unsignedBitsRequired(maxDelta);
      DirectWriter writer = DirectWriter.getInstance(data, bufferSize, bitsRequired);
      for (int i = 0; i < bufferSize; ++i) {
        writer.add(buffer[i]);
      }
      writer.finish();
      meta.writeByte((byte) bitsRequired);
    }
    bufferSize = 0;
  }
  public void add(long v) throws IOException {
    if (v < previous) { // 如果不是遞增的,則不應(yīng)該使用DirectMonotonicWriter
      throw new IllegalArgumentException("Values do not come in order: " + previous + ", " + v);
    }
    if (bufferSize == buffer.length) { // buffer滿了
      flush();
    }
    buffer[bufferSize++] = v;
    previous = v;
    count++;
  }
  public void finish() throws IOException {
    if (count != numValues) {
      throw new IllegalStateException(
          "Wrong number of values added, expected: " + numValues + ", got: " + count);
    }
    if (finished) {
      throw new IllegalStateException("#finish has been called already");
    }
    if (bufferSize > 0) { // 如果還有數(shù)據(jù),則先編碼
      flush();
    }
    finished = true;
  }
}

總結(jié)

之前在看索引文件構(gòu)建的源碼時(shí),會(huì)經(jīng)常碰到對(duì)于正整數(shù)的批量壓縮應(yīng)用,而且有好幾個(gè)不同的類,當(dāng)時(shí)都是把它們當(dāng)成黑盒忽略,現(xiàn)在花了點(diǎn)時(shí)間,總算是把這些不同應(yīng)用場(chǎng)景都區(qū)分了。本文沒(méi)有羅列Lucene中全部的編碼壓縮應(yīng)用,后面如果有再碰到影響我看源碼的壓縮相關(guān)的,我會(huì)補(bǔ)充進(jìn)來(lái)。

以上就是Lucene源碼系列多值編碼壓縮算法實(shí)例詳解的詳細(xì)內(nèi)容,更多關(guān)于Lucene多值編碼壓縮算法的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • java弱口令檢測(cè)機(jī)制解析

    java弱口令檢測(cè)機(jī)制解析

    這篇文章主要介紹了java弱口令檢測(cè)機(jī)制,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-09-09
  • JSON.toJSONString()方法在Java中的使用方法及應(yīng)用場(chǎng)景

    JSON.toJSONString()方法在Java中的使用方法及應(yīng)用場(chǎng)景

    這篇文章主要給大家介紹了關(guān)于JSON.toJSONString()方法在Java中的使用方法及應(yīng)用場(chǎng)景,JSON.toJSONString是將對(duì)象轉(zhuǎn)化為Json字符串,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2024-04-04
  • Springmvc應(yīng)用Mongodb分頁(yè)實(shí)現(xiàn)

    Springmvc應(yīng)用Mongodb分頁(yè)實(shí)現(xiàn)

    這篇文章主要為大家詳細(xì)介紹了Springmvc應(yīng)用Mongodb分頁(yè)實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-11-11
  • spring中WebClient如何設(shè)置連接超時(shí)時(shí)間以及讀取超時(shí)時(shí)間

    spring中WebClient如何設(shè)置連接超時(shí)時(shí)間以及讀取超時(shí)時(shí)間

    這篇文章主要給大家介紹了關(guān)于spring中WebClient如何設(shè)置連接超時(shí)時(shí)間以及讀取超時(shí)時(shí)間的相關(guān)資料,WebClient是Spring框架5.0引入的基于響應(yīng)式編程模型的HTTP客戶端,它提供一種簡(jiǎn)便的方式來(lái)處理HTTP請(qǐng)求和響應(yīng),需要的朋友可以參考下
    2024-08-08
  • Java實(shí)現(xiàn)并查集

    Java實(shí)現(xiàn)并查集

    這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)并查集,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-07-07
  • 解決Nacos成功啟動(dòng)但是無(wú)法訪問(wèn) (Connection refused)

    解決Nacos成功啟動(dòng)但是無(wú)法訪問(wèn) (Connection refused)

    這篇文章主要介紹了解決Nacos成功啟動(dòng)但是無(wú)法訪問(wèn) (Connection refused)問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-06-06
  • Java利用反射如何查找使用指定注解的類詳解

    Java利用反射如何查找使用指定注解的類詳解

    這篇文章主要給大家介紹了關(guān)于Java利用反射如何查找使用指定注解的類的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。
    2017-09-09
  • JAVA十大排序算法之堆排序詳解

    JAVA十大排序算法之堆排序詳解

    這篇文章主要介紹了java中的冒泡排序,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考
    2021-08-08
  • mybatis3使用@Select等注解實(shí)現(xiàn)增刪改查操作

    mybatis3使用@Select等注解實(shí)現(xiàn)增刪改查操作

    這篇文章主要介紹了mybatis3使用@Select等注解實(shí)現(xiàn)增刪改查操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-11-11
  • 解決mybatis plus報(bào)錯(cuò)Invalid bound statement (not found):問(wèn)題

    解決mybatis plus報(bào)錯(cuò)Invalid bound statement 

    在使用MyBatis時(shí)遇到InvalidBoundStatement異常,常因多個(gè)MapperScan配置沖突或者包掃描路徑設(shè)置錯(cuò)誤,解決方法包括保留一個(gè)MapperScan聲明、檢查jar包沖突、確保命名空間和掃描路徑正確,使用@TableId注解指定主鍵
    2024-11-11

最新評(píng)論

色婷婷综合激情五月免费观看| 超级碰碰在线视频免费观看| 国产午夜亚洲精品不卡在线观看| 操人妻嗷嗷叫视频一区二区| 国产日本欧美亚洲精品视| 影音先锋女人av噜噜色| huangse网站在线观看| 欧美精品中文字幕久久二区| 国产第一美女一区二区三区四区| 少妇深喉口爆吞精韩国| 经典av尤物一区二区| 午夜激情高清在线观看| 亚洲精品国产在线电影| 国产久久久精品毛片| 内射久久久久综合网| 精品久久婷婷免费视频| 97人人模人人爽人人喊| 直接观看免费黄网站| 夏目彩春在线中文字幕| 午夜频道成人在线91| 久久久久国产成人精品亚洲午夜| 另类av十亚洲av| 亚洲高清国产自产av| 伊人精品福利综合导航| 一区二区三区精品日本| 中文字幕最新久久久| 人人妻人人爱人人草| 女同久久精品秋霞网| 欧美一区二区三区四区性视频| 午夜免费体验区在线观看| 521精品视频在线观看| 在线视频这里只有精品自拍| 精品一区二区三四区| 美女日逼视频免费观看| 中文字幕中文字幕人妻| 亚洲av色图18p| 性欧美日本大妈母与子| 精品人妻每日一部精品| 日本高清成人一区二区三区| 一级黄片大鸡巴插入美女| 日本熟女50视频免费| 中文字幕1卡1区2区3区| 中文字幕在线观看极品视频| 欧美偷拍亚洲一区二区| 男生舔女生逼逼视频| 国产精品成久久久久三级蜜臀av| 午夜美女福利小视频| 日本人妻少妇18—xx| 51国产成人精品视频| 欧美视频中文一区二区三区| 日本特级片中文字幕| 日本人妻精品久久久久久| 精品欧美一区二区vr在线观看| 成人av免费不卡在线观看| 国产av欧美精品高潮网站| 啊啊好慢点插舔我逼啊啊啊视频| 中文亚洲欧美日韩无线码| 午夜精品一区二区三区4| 日本最新一二三区不卡在线| 日韩a级精品一区二区| 2022中文字幕在线| 亚洲av色香蕉一区二区三区| 偷拍3456eee| 熟女91pooyn熟女| xxx日本hd高清| 国产精品人妻一区二区三区网站| 一区二区视频在线观看视频在线| 亚洲 中文字幕在线 日韩| 国产在线拍揄自揄视频网站| 国产在线免费观看成人| 欧美中文字幕一区最新网址| 天堂中文字幕翔田av| av完全免费在线观看av| 好吊操视频这里只有精品| 欧洲国产成人精品91铁牛tv| 熟女在线视频一区二区三区| 国产在线拍揄自揄视频网站| 啊用力插好舒服视频| 中文字幕人妻一区二区视频| 午夜影院在线观看视频羞羞羞| 欧美怡红院视频在线观看| 国产又粗又猛又爽又黄的视频美国| av中文字幕在线观看第三页| 91大屁股国产一区二区| 国产成人无码精品久久久电影| 天天射,天天操,天天说| 精品一区二区三区在线观看| 国产一区二区火爆视频| 一二三区在线观看视频| 国产精品一区二区三区蜜臀av | 国产精品一区二区久久久av| 综合激情网激情五月五月婷婷| 人妻少妇一区二区三区蜜桃| 直接观看免费黄网站| 999九九久久久精品| 久久久久久cao我的性感人妻| 精品国产污污免费网站入口自| 亚洲图库另类图片区| 9l人妻人人爽人人爽| 亚洲av男人天堂久久| 天天操天天爽天天干| 欧美日韩熟女一区二区三区| 淫秽激情视频免费观看| 国产精品久久久黄网站| 国产精品人妻66p| 中国视频一区二区三区| 色哟哟在线网站入口| 亚洲变态另类色图天堂网| 18禁精品网站久久| 日本av熟女在线视频| 国产亚洲欧美视频网站| 动漫黑丝美女的鸡巴| 韩国爱爱视频中文字幕| 男人的天堂在线黄色| 精品一区二区三四区| 国产日韩一区二区在线看| 手机看片福利盒子日韩在线播放| 成年午夜免费无码区| 一色桃子久久精品亚洲 | 中文字幕人妻av在线观看| 欧美一区二区三区激情啪啪啪| 成年人黄色片免费网站| 国产麻豆国语对白露脸剧情| 日韩人妻在线视频免费| 超碰97人人澡人人| 亚洲欧美福利在线观看| 毛茸茸的大外阴中国视频| 在线观看国产网站资源| 亚洲1069综合男同| 日韩a级黄色小视频| 黑人变态深video特大巨大| 美女吃鸡巴操逼高潮视频| 精产国品久久一二三产区区别| 最后99天全集在线观看| 亚洲第17页国产精品| 欧洲日韩亚洲一区二区三区| 天天日夜夜干天天操| 中文字幕在线观看极品视频| 91精品一区二区三区站长推荐| 日本少妇精品免费视频| 啪啪啪啪啪啪啪啪av| 中国把吊插入阴蒂的视频| 天天艹天天干天天操| 国产精品视频一区在线播放| 天美传媒mv视频在线观看| 国产日韩av一区二区在线| 黄色视频成年人免费观看| 亚洲欧美成人综合视频| 欧亚乱色一区二区三区| 91欧美在线免费观看| 天天操天天弄天天射| 91啪国自产中文字幕在线| 精品日产卡一卡二卡国色天香| 手机看片福利盒子日韩在线播放| 一区二区视频在线观看免费观看| 天天日天天日天天射天天干| 沈阳熟妇28厘米大战黑人| 天天干天天爱天天色| 一级A一级a爰片免费免会员| 亚洲 欧美 自拍 偷拍 在线| 91chinese在线视频| 亚洲福利精品视频在线免费观看| 国产妇女自拍区在线观看| 亚洲成人三级在线播放| 日韩一区二区三区三州| 久久热这里这里只有精品| 99精品一区二区三区的区| 亚洲国产精品久久久久久6| 在线播放一区二区三区Av无码| 18禁美女羞羞免费网站| 精品一区二区亚洲欧美| 999热精品视频在线| 日韩av免费观看一区| 狠狠的往里顶撞h百合| 国产美女午夜福利久久| 热久久只有这里有精品| 97年大学生大白天操逼| 日韩中文字幕福利av| 91 亚洲视频在线观看| 亚洲一区二区三区五区| 亚洲图库另类图片区| 欧美亚洲牲夜夜综合久久| 久草极品美女视频在线观看| 精品亚洲中文字幕av| 大鸡八强奸视频在线观看| 青青草国内在线视频精选| 国产精彩对白一区二区三区 | 免费在线观看污污视频网站| 人人妻人人爱人人草| 在线免费观看亚洲精品电影| 久久久久久久久久性潮| 91成人精品亚洲国产| 亚洲伊人久久精品影院一美女洗澡 | 亚洲人妻国产精品综合| 亚洲av人人澡人人爽人人爱| 色婷婷精品大在线观看| 高清成人av一区三区| 亚洲av日韩高清hd| 亚洲av色香蕉一区二区三区 | 美女吃鸡巴操逼高潮视频| 午夜婷婷在线观看视频| 亚洲一区二区三区在线高清| 国产亚洲欧美45p| 欧美偷拍亚洲一区二区| 成年人免费看在线视频| 婷婷久久久综合中文字幕| 亚洲午夜伦理视频在线| 38av一区二区三区| 久久尻中国美女视频| 青娱乐极品视频青青草| 不卡一区一区三区在线| 偷青青国产精品青青在线观看 | 少妇人妻二三区视频| 亚洲av自拍天堂网| 熟女91pooyn熟女| 熟女在线视频一区二区三区| 国产福利小视频大全| 91亚洲精品干熟女蜜桃频道| 亚洲成人情色电影在线观看| 久久久久久9999久久久久| 亚洲成人av在线一区二区| aⅴ精产国品一二三产品| 农村胖女人操逼视频| 春色激情网欧美成人| 免费在线观看污污视频网站| 51国产成人精品视频| 国产一区二区久久久裸臀| 老司机深夜免费福利视频在线观看| 亚洲的电影一区二区三区| 色婷婷精品大在线观看| 青青草国内在线视频精选| 免费国产性生活视频| 国产精品久久久久网| 国产精品人妻一区二区三区网站 | 国产免费高清视频视频| 日韩人妻在线视频免费| 亚洲午夜伦理视频在线| 国产揄拍高清国内精品对白| 久久精品国产亚洲精品166m| 日韩成人免费电影二区| 国产一级精品综合av| 在线观看的黄色免费网站| 国产精品自拍在线视频| 日本一区美女福利视频| 91桃色成人网络在线观看| 欧美香蕉人妻精品一区二区| 阿v天堂2014 一区亚洲| 久久这里只有精品热视频| 人妻最新视频在线免费观看| 91精品综合久久久久3d动漫| 中文字幕中文字幕 亚洲国产| 老司机深夜免费福利视频在线观看| 人妻少妇亚洲精品中文字幕| av久久精品北条麻妃av观看| 狠狠操操操操操操操操操| 综合国产成人在线观看| 亚洲伊人久久精品影院一美女洗澡| 337p日本大胆欧美人| 国产大学生援交正在播放| 超pen在线观看视频公开97| 熟女人妻在线观看视频| 国内精品在线播放第一页| 亚洲午夜福利中文乱码字幕| 国产av福利网址大全| 天天操天天弄天天射| 日本丰满熟妇大屁股久久| 又粗又长 明星操逼小视频 | 亚洲熟妇无码一区二区三区| 搡老妇人老女人老熟女| chinese国产盗摄一区二区 | 成人久久精品一区二区三区| 日本熟女精品一区二区三区| 欧美日本国产自视大全| 欧美3p在线观看一区二区三区| 精品久久久久久久久久久久人妻| 亚洲av无码成人精品区辽| 伊人情人综合成人久久网小说| 色狠狠av线不卡香蕉一区二区 | 熟女国产一区亚洲中文字幕| 天天操天天操天天碰| 啪啪啪啪啪啪啪啪av| 天天干天天爱天天色| 欧美成一区二区三区四区| 偷拍自拍亚洲视频在线观看| 精品国产污污免费网站入口自| 欧美亚洲牲夜夜综合久久| 91国内精品久久久久精品一| 91中文字幕免费在线观看| 久久精品国产999| 91‖亚洲‖国产熟女| 在线观看操大逼视频| 国产美女精品福利在线| 天堂av在线播放免费| 国产精品伦理片一区二区| 香蕉片在线观看av| 成人av电影免费版| 亚洲av无乱一区二区三区性色| 538精品在线观看视频| 风流唐伯虎电视剧在线观看| 人妻另类专区欧美制服| 国产精品欧美日韩区二区 | 欧美色呦呦最新网址| 黄色大片男人操女人逼| 99精品国产自在现线观看| 亚洲av自拍天堂网| 在线视频免费观看网| 天天做天天爽夜夜做少妇| 青青青视频手机在线观看| 人妻无码中文字幕专区| 欧洲亚洲欧美日韩综合| 97青青青手机在线视频| 婷婷久久久综合中文字幕| 午夜久久久久久久精品熟女| 成人国产小视频在线观看| 中文字幕高清资源站| 麻豆性色视频在线观看| 精品美女在线观看视频在线观看| 亚洲精品无码色午夜福利理论片| 中文字幕 码 在线视频| 国产午夜无码福利在线看| 日本美女成人在线视频| 97a片免费在线观看| 亚洲伊人av天堂有码在线| 国产伦精品一区二区三区竹菊| 国产精品国产三级国产午| 91人妻精品一区二区在线看| 馒头大胆亚洲一区二区| 亚洲自拍偷拍精品网| 天天操天天干天天艹| 亚洲中文精品人人免费| 久草视频 久草视频2| 日韩黄色片在线观看网站| 久久久久久久久久一区二区三区| 春色激情网欧美成人| 国产精品自拍在线视频| 宅男噜噜噜666国产| 午夜免费观看精品视频| 啊啊啊想要被插进去视频| 免费一级特黄特色大片在线观看 | 国产精品日韩欧美一区二区| 国产91精品拍在线观看| 国产成人精品久久二区91| 精品一区二区三区欧美| 国产麻豆乱子伦午夜视频观看| 新婚人妻聚会被中出| 国产精品人久久久久久| 国产精彩福利精品视频| 亚洲国产精品免费在线观看| 少妇高潮一区二区三区| 免费十精品十国产网站| 国产精品手机在线看片| 人妻熟女在线一区二区| 亚洲图片偷拍自拍区| 国产亚洲国产av网站在线| 国产va精品免费观看| 日本www中文字幕| 婷婷五月亚洲综合在线| 国产妇女自拍区在线观看| 免费在线看的黄网站| 色狠狠av线不卡香蕉一区二区| 免费看国产av网站| 国产一级麻豆精品免费| 在线观看视频网站麻豆| 一区二区三区日韩久久| 亚洲免费福利一区二区三区| 天堂av中文在线最新版| 狠狠操狠狠操免费视频| 欧美偷拍亚洲一区二区| 97人人妻人人澡人人爽人人精品| 男人的天堂av日韩亚洲| 国产精品午夜国产小视频| 2025年人妻中文字幕乱码在线 | 又色又爽又黄的美女裸体| 激情色图一区二区三区| 国产老熟女伦老熟妇ⅹ| 天天做天天干天天舔| 青青青视频手机在线观看| 亚洲av无码成人精品区辽| 精品国产污污免费网站入口自| 欧美黑人性暴力猛交喷水| 亚洲激情偷拍一区二区| 午夜精品一区二区三区更新| 中文乱理伦片在线观看| 欧美天堂av无线av欧美| 日本熟女精品一区二区三区| 男人天堂最新地址av| 91p0rny九色露脸熟女| 久久麻豆亚洲精品av| 国产视频一区二区午夜| caoporm超碰国产| 女同性ⅹxx女同h偷拍| 天天干狠狠干天天操| 人妻3p真实偷拍一二区| 最新中文字幕乱码在线| 偷青青国产精品青青在线观看| 久精品人妻一区二区三区| 日本又色又爽又黄又粗| 国产在线观看黄色视频| 国产在线91观看免费观看| 国产欧美精品免费观看视频| 天天干天天爱天天色| 午夜精品久久久久久99热| 久久农村老妇乱69系列| 自拍偷拍,中文字幕| 日韩人妻丝袜中文字幕| 久久精品在线观看一区二区| 2o22av在线视频| 在线观看欧美黄片一区二区三区| 亚洲av成人免费网站| 91精品综合久久久久3d动漫| 精品国产在线手机在线| 中文字幕日韩精品日本| 欧美aa一级一区三区四区 | 国产亚州色婷婷久久99精品| 精品久久久久久久久久久a√国产 日本女大学生的黄色小视频 | 成年人黄视频在线观看| 亚洲福利天堂久久久久久| 国产乱子伦一二三区| 18禁精品网站久久| 国产精品久久久久久美女校花| 欧美精品欧美极品欧美视频| 日韩伦理短片在线观看| 亚洲熟妇无码一区二区三区| 亚洲欧美人精品高清| 日日夜夜精品一二三| 老司机免费视频网站在线看| 人妻熟女在线一区二区| 国产麻豆精品人妻av| 91综合久久亚洲综合| 久久久久久久久久久久久97| 日本少妇的秘密免费视频| 天天日天天添天天爽| 中出中文字幕在线观看| 久久久久久久精品成人热| 激情五月婷婷免费视频| 午夜福利人人妻人人澡人人爽| 黄色资源视频网站日韩| 青青青爽视频在线播放| 精品区一区二区三区四区人妻| 亚洲福利午夜久久久精品电影网| 91成人在线观看免费视频| 色av色婷婷人妻久久久精品高清| av中文字幕电影在线看| 国产精品系列在线观看一区二区 | 色伦色伦777国产精品| 淫秽激情视频免费观看| 青青社区2国产视频| 伊人情人综合成人久久网小说| 久草极品美女视频在线观看| 日韩av有码中文字幕| 国产极品美女久久久久久| 蜜臀成人av在线播放| 日韩影片一区二区三区不卡免费 | 成人区人妻精品一区二视频| 欧美偷拍亚洲一区二区| 888欧美视频在线| 欧美一级视频一区二区| 直接观看免费黄网站| 91色老99久久九九爱精品| 传媒在线播放国产精品一区| 啊用力插好舒服视频| 美女小视频网站在线| 啊啊好大好爽啊啊操我啊啊视频 | 成人影片高清在线观看| 夜夜操,天天操,狠狠操| 婷婷午夜国产精品久久久| 日韩欧美高清免费在线| 熟女在线视频一区二区三区| av在线shipin| 91大神福利视频网| 天天操天天操天天碰| 精品一区二区三区三区色爱| 日本少妇精品免费视频| 色偷偷伊人大杳蕉综合网| av在线播放国产不卡| 国产成人精品一区在线观看| 日韩av免费观看一区| 国产清纯美女al在线| 蜜桃视频入口久久久| 婷婷色国产黑丝少妇勾搭AV| 在线播放国产黄色av| 66久久久久久久久久久| 天天日天天天天天天天天天天| 午夜精品福利一区二区三区p| 搡老妇人老女人老熟女| 国产精品国产三级国产午| 性欧美激情久久久久久久| 日韩美av高清在线| 欧美综合婷婷欧美综合| 国产福利小视频二区| 欧美偷拍自拍色图片| 久久热久久视频在线观看| 亚洲精品亚洲人成在线导航| 懂色av之国产精品| 亚洲av色香蕉一区二区三区| av资源中文字幕在线观看| 亚洲一区av中文字幕在线观看| 国产亚州色婷婷久久99精品| 中文字幕日韩无敌亚洲精品| 中文字母永久播放1区2区3区| 久久久久久久精品成人热| 2022天天干天天操| 婷婷午夜国产精品久久久| 一区二区麻豆传媒黄片| 2021天天色天天干| 国产黄色a级三级三级三级| 久久久久久久精品老熟妇| 在线观看一区二区三级| 人妻少妇性色欲欧美日韩| 久久麻豆亚洲精品av| 亚洲欧美综合另类13p| 动漫黑丝美女的鸡巴| 国产在线观看黄色视频| 亚洲av日韩精品久久久| 天天日天天爽天天爽| av中文字幕福利网| 成年人免费看在线视频| 人妻激情图片视频小说| 亚洲精品午夜久久久久| 婷婷六月天中文字幕| 亚洲av第国产精品| 日日操综合成人av| 欧美精品亚洲精品日韩在线| 亚洲人人妻一区二区三区| 久久久久久久久久久久久97| 欧美老妇精品另类不卡片| 一区二区三区蜜臀在线| 国产精品伦理片一区二区| 久精品人妻一区二区三区 | 青青青国产免费视频| 国产黄网站在线观看播放| 伊人成人综合开心网| 男人天堂最新地址av| 99热这里只有国产精品6| 黄色成年网站午夜在线观看| 男女第一次视频在线观看| 亚洲 中文字幕在线 日韩| gav成人免费播放| 日韩午夜福利精品试看| 综合激情网激情五月天| 极品丝袜一区二区三区| 插小穴高清无码中文字幕| 国产精品黄大片在线播放| 亚洲欧美一区二区三区爱爱动图| 美女少妇亚洲精选av| 97人妻人人澡爽人人精品| 国产va在线观看精品| 一区二区三区日本伦理| 伊人成人综合开心网| 欧美va不卡视频在线观看| 日本黄在免费看视频| 香蕉aⅴ一区二区三区| 色在线观看视频免费的| 麻豆性色视频在线观看| 91啪国自产中文字幕在线| 在线观看黄色成年人网站| 黄色男人的天堂视频| 亚洲少妇高潮免费观看| 亚洲 自拍 色综合图| 色97视频在线播放| 动漫精品视频在线观看| 中国黄色av一级片| 精品首页在线观看视频| av新中文天堂在线网址| 久久热这里这里只有精品| 久草极品美女视频在线观看| 18禁网站一区二区三区四区| 欧美精品中文字幕久久二区| 91一区精品在线观看| 黄色大片男人操女人逼| 这里只有精品双飞在线播放| 欧亚乱色一区二区三区| 夜夜骑夜夜操夜夜奸| 国产三级精品三级在线不卡| 在线免费观看视频一二区| 少妇一区二区三区久久久| 九九视频在线精品播放| 久久久久91精品推荐99| 91天堂天天日天天操| 大香蕉大香蕉大香蕉大香蕉大香蕉 | 成年午夜影片国产片| 99国内小视频在现欢看| 91高清成人在线视频| 亚洲欧美福利在线观看| 中文字幕免费在线免费| 天天草天天色天天干| 国产精品久久9999| 99久久成人日韩欧美精品| 在线观看国产免费麻豆| 特大黑人巨大xxxx| 亚洲中文精品人人免费| 国产福利小视频大全| 婷婷久久久综合中文字幕| 日韩美av高清在线| 亚洲1区2区3区精华液| 视频一区 二区 三区 综合| 97国产福利小视频合集| 亚洲国产在线精品国偷产拍| 亚洲福利精品视频在线免费观看 | 国产高清在线观看1区2区| 51国产成人精品视频| 青青青青操在线观看免费| 国产精品黄色的av| 岛国一区二区三区视频在线| 亚洲精品国品乱码久久久久| xxx日本hd高清| 亚洲精品亚洲人成在线导航| 美女张开两腿让男人桶av| 一区二区免费高清黄色视频| wwwxxx一级黄色片| 99久久久无码国产精品性出奶水| 日本中文字幕一二区视频| 丝袜肉丝一区二区三区四区在线| 亚洲精品在线资源站| 青青青国产片免费观看视频| 欧美精品国产综合久久| 五色婷婷综合狠狠爱| 中文字幕之无码色多多| 日本欧美视频在线观看三区| 日本真人性生活视频免费看| 欧美在线一二三视频| 激情小视频国产在线| 早川濑里奈av黑人番号| 国产精彩对白一区二区三区| 亚洲午夜高清在线观看| 日本熟女50视频免费| 国产精品视频欧美一区二区| 伊人情人综合成人久久网小说| 青青青青青免费视频| 亚洲av成人免费网站| 日日操综合成人av| 精品国产污污免费网站入口自| 精品suv一区二区69| 激情五月婷婷免费视频| 日本熟妇色熟妇在线观看| 人人妻人人爱人人草| 丝袜国产专区在线观看| 少妇与子乱在线观看| 人妻无码色噜噜狠狠狠狠色| 888亚洲欧美国产va在线播放| 亚洲精品三级av在线免费观看| 熟女妇女老妇一二三区| 国产大学生援交正在播放| 久久99久久99精品影院| 日韩精品中文字幕福利| 亚洲av日韩高清hd| 2022精品久久久久久中文字幕| 污污小视频91在线观看| 啊啊好慢点插舔我逼啊啊啊视频| 美女大bxxxx内射| 66久久久久久久久久久| 欧美专区日韩专区国产专区| 人妻爱爱 中文字幕| 天天操天天弄天天射| 亚洲福利精品视频在线免费观看| 天天干天天日天天谢综合156| 欧美亚洲一二三区蜜臀| 亚洲欧美国产麻豆综合| 这里只有精品双飞在线播放| 国产亚洲精品品视频在线| 啊啊啊想要被插进去视频| 国产亚洲视频在线二区| 亚洲av在线观看尤物| 久久尻中国美女视频| 少妇ww搡性bbb91| 福利片区一区二体验区| 色综合色综合色综合色| 2020久久躁狠狠躁夜夜躁| 2020国产在线不卡视频| 97精品成人一区二区三区 | 免费岛国喷水视频在线观看| 天码人妻一区二区三区在线看| 任你操视频免费在线观看| 春色激情网欧美成人| 久久美欧人妻少妇一区二区三区| 男人和女人激情视频| av一本二本在线观看| 免费在线播放a级片| 国产+亚洲+欧美+另类| 绝顶痉挛大潮喷高潮无码 | 香蕉片在线观看av| 岛国av高清在线成人在线| 色婷婷六月亚洲综合香蕉| 插小穴高清无码中文字幕| 天天日天天干天天舔天天射| 亚洲美女高潮喷浆视频| 性色蜜臀av一区二区三区| 快插进小逼里大鸡吧视频| 最新欧美一二三视频| 人妻激情图片视频小说| 99热这里只有国产精品6| 久精品人妻一区二区三区| gav成人免费播放| 色综合久久无码中文字幕波多| 最后99天全集在线观看| 2020国产在线不卡视频| 亚洲人成精品久久久久久久| 一区二区三区激情在线| 性色蜜臀av一区二区三区| 一区二区麻豆传媒黄片| 亚洲欧美日韩视频免费观看| 大胸性感美女羞爽操逼毛片| 一级A一级a爰片免费免会员| 唐人色亚洲av嫩草| 亚洲嫩模一区二区三区| asmr福利视频在线观看| 午夜美女福利小视频| 国产精品中文av在线播放| 做爰视频毛片下载蜜桃视频1| 伊人网中文字幕在线视频| 91精品一区二区三区站长推荐| 欧美精产国品一二三区| 男人和女人激情视频| 只有精品亚洲视频在线观看| 欧美韩国日本国产亚洲| 亚洲国产精品中文字幕网站| 国产亚洲欧美视频网站| 天天干天天插天天谢| 区一区二区三国产中文字幕| 成人24小时免费视频| 男生用鸡操女生视频动漫| av无限看熟女人妻另类av| 亚洲图片欧美校园春色| 亚洲午夜伦理视频在线| 岳太深了紧紧的中文字幕| 日本女大学生的黄色小视频| 国产伊人免费在线播放| 日本成人一区二区不卡免费在线| 黄色成人在线中文字幕| 肏插流水妹子在线乐播下载| 青娱乐蜜桃臀av色| 色狠狠av线不卡香蕉一区二区 | 青青青视频自偷自拍38碰| 亚洲国际青青操综合网站| 亚洲护士一区二区三区| 在线观看国产网站资源| 美洲精品一二三产区区别| 99热99re在线播放| 欧美日韩亚洲国产无线码| 男女第一次视频在线观看| 欧美日韩激情啪啪啪| 成人免费公开视频无毒| 日本一本午夜在线播放| 色偷偷伊人大杳蕉综合网| 国产精品人久久久久久| 在线观看免费av网址大全| 小泽玛利亚视频在线观看| 亚洲午夜电影之麻豆| 欧美黑人与人妻精品| 成人午夜电影在线观看 久久| 在线视频国产欧美日韩| 一级黄色片夫妻性生活| 国际av大片在线免费观看| 51国产偷自视频在线播放| 91中文字幕免费在线观看| 人人妻人人爽人人澡人人精品| 一个人免费在线观看ww视频| 在线免费观看亚洲精品电影| 边摸边做超爽毛片18禁色戒| 亚洲一级av无码一级久久精品| 亚洲人一区二区中文字幕| 久久久精品999精品日本| 国产精品手机在线看片| 少妇高潮一区二区三区| 成年美女黄网站18禁久久| 欧美亚洲偷拍自拍色图| 中文字幕日韩精品日本| 天堂va蜜桃一区入口| 一区二区三区精品日本| 日本少妇人妻xxxxxhd| 欧美日韩在线精品一区二区三| 国产一级麻豆精品免费| 88成人免费av网站| 欧美视频不卡一区四区| v888av在线观看视频| 国产福利小视频二区| 91精品国产高清自在线看香蕉网 | av在线观看网址av| 鸡巴操逼一级黄色气| 岳太深了紧紧的中文字幕| okirakuhuhu在线观看| 大黑人性xxxxbbbb| 日本一区二区三区免费小视频| av线天堂在线观看| av在线shipin| 国产综合高清在线观看| 一区二区免费高清黄色视频| 视频一区 二区 三区 综合| 黄色男人的天堂视频| 91国内视频在线观看| 国产黄色a级三级三级三级| 亚洲激情偷拍一区二区| 不卡一不卡二不卡三| japanese日本熟妇另类| 扒开让我视频在线观看| av资源中文字幕在线观看| 99视频精品全部15| 欧美精品免费aaaaaa| 久久一区二区三区人妻欧美| 99久久超碰人妻国产| 精品亚洲中文字幕av| 欧美 亚洲 另类综合| 91久久人澡人人添人人爽乱| 老师让我插进去69AV| 久久精品亚洲国产av香蕉| 国产高清女主播在线| 9国产精品久久久久老师| 国产av一区2区3区| 国产精品入口麻豆啊啊啊| 成年人黄色片免费网站| 黄色三级网站免费下载| 91精品国产91久久自产久强| 在线免费观看黄页视频| 视频二区在线视频观看| 久久精品在线观看一区二区| 2017亚洲男人天堂| 91麻豆精品91久久久久同性| 人妻另类专区欧美制服| rct470中文字幕在线| 亚洲 中文 自拍 另类 欧美| 97少妇精品在线观看| 精品乱子伦一区二区三区免费播| 国产无遮挡裸体免费直播视频| 美女福利视频导航网站| 真实国模和老外性视频| 五月精品丁香久久久久福利社| 国产成人一区二区三区电影网站 | 日本a级视频老女人| 天堂av中文在线最新版| 在线亚洲天堂色播av电影| 亚洲高清视频在线不卡| 亚洲 中文 自拍 无码| 国产亚洲成人免费在线观看| 国产又粗又硬又猛的毛片视频 | 香蕉片在线观看av| 91社福利《在线观看| 超碰在线中文字幕一区二区| 超碰中文字幕免费观看| 免费在线福利小视频| 青青青青青青草国产| 麻豆性色视频在线观看| 日韩欧美一级黄片亚洲| 2020韩国午夜女主播在线| 91久久综合男人天堂| 亚洲欧美成人综合在线观看| 爱爱免费在线观看视频| 日本人妻少妇18—xx| 亚洲一级 片内射视正片| 国产成人一区二区三区电影网站| 亚洲1区2区3区精华液| 欧美老妇精品另类不卡片| 日本免费午夜视频网站| 97a片免费在线观看| 2020中文字幕在线播放| 天天操天天干天天插| yy6080国产在线视频| 91av精品视频在线| 国际av大片在线免费观看| 天堂女人av一区二区| 成人精品视频99第一页| 亚洲视频在线视频看视频在线| 亚洲精品色在线观看视频| 亚洲一区二区三区精品视频在线 | 国产精品熟女久久久久浪潮| 成人高清在线观看视频| 在线免费观看日本伦理| av黄色成人在线观看| huangse网站在线观看| 又黄又刺激的午夜小视频| 美女操逼免费短视频下载链接| av天堂加勒比在线| 国产精品熟女久久久久浪潮| 中国视频一区二区三区| 亚洲青青操骚货在线视频| 久草视频首页在线观看| 91p0rny九色露脸熟女| 5528327男人天堂| 秋霞午夜av福利经典影视| 日本一道二三区视频久久 | 午夜免费体验区在线观看| 99久久超碰人妻国产| 亚洲激情偷拍一区二区| 91成人在线观看免费视频| avjpm亚洲伊人久久| 毛片一级完整版免费| 在线观看的黄色免费网站| av在线免费中文字幕| 摧残蹂躏av一二三区| 国产精品亚洲在线观看| 东京热男人的av天堂| 黄色大片免费观看网站| 国产精品三级三级三级| 在线观看av2025| 亚洲欧美自拍另类图片| 亚洲免费福利一区二区三区| 福利国产视频在线观看| 夜女神免费福利视频| 国产九色91在线视频| 欧美精品中文字幕久久二区| 欧美女同性恋免费a| 天码人妻一区二区三区在线看| 91亚洲国产成人精品性色| 青青青国产免费视频| 国产va在线观看精品| 精品国产亚洲av一淫| 欧美国产亚洲中英文字幕| 老司机深夜免费福利视频在线观看| 色吉吉影音天天干天天操| 日本精品一区二区三区在线视频。| 日韩一区二区电国产精品| 国产性色生活片毛片春晓精品| chinese国产盗摄一区二区| 男人天堂av天天操| 风流唐伯虎电视剧在线观看| 午夜在线精品偷拍一区二| 日韩北条麻妃一区在线| 中国熟女@视频91| 亚洲伊人久久精品影院一美女洗澡| 欧美久久一区二区伊人| 久久久久久久久久久久久97| 天天干夜夜操啊啊啊| 天天艹天天干天天操| 精产国品久久一二三产区区别| 国产性色生活片毛片春晓精品 | 欧亚日韩一区二区三区观看视频| 特级欧美插插插插插bbbbb| 久草视频在线看免费| 亚洲成人三级在线播放| 国产97在线视频观看| 国产无遮挡裸体免费直播视频| 97精品综合久久在线| 免费岛国喷水视频在线观看 | 激情五月婷婷免费视频| 97青青青手机在线视频| 亚洲熟妇无码一区二区三区| 99婷婷在线观看视频| 亚洲 中文 自拍 另类 欧美| 亚洲av无女神免非久久| 被大鸡吧操的好舒服视频免费| 一区二区三区日韩久久| 亚洲午夜电影之麻豆| 国产精品三级三级三级| 97少妇精品在线观看| 天堂av在线播放免费| 69精品视频一区二区在线观看| 亚洲av男人的天堂你懂的| 国产精品国色综合久久| 亚洲最大黄了色网站| 国产中文字幕四区在线观看| 久久久久久久精品成人热| 91色秘乱一区二区三区| 亚洲一区二区久久久人妻| 国产视频网站一区二区三区 | 天天干夜夜操啊啊啊| 天堂v男人视频在线观看| 大香蕉伊人国产在线| www日韩毛片av| 亚洲成人线上免费视频观看| 亚洲欧美激情中文字幕| 91人妻人人做人人爽在线| aiss午夜免费视频| 黄色中文字幕在线播放| 国产精品国色综合久久| 含骚鸡巴玩逼逼视频| av中文字幕电影在线看| 国产福利小视频二区| 国产自拍黄片在线观看| h国产小视频福利在线观看| 国产精彩福利精品视频| 亚洲成人av一区在线| 国产女人被做到高潮免费视频| 人妻凌辱欧美丰满熟妇| 97超碰国语国产97超碰| 动漫美女的小穴视频| 最新欧美一二三视频| 91久久人澡人人添人人爽乱| 亚洲av黄色在线网站| 亚洲免费福利一区二区三区| 老司机午夜精品视频资源| 美女福利视频网址导航| 日本一道二三区视频久久| 扒开让我视频在线观看| 欧美久久久久久三级网| 天天日天天干天天舔天天射| 一区二区在线视频中文字幕| 日本三极片视频网站观看| 国产普通话插插视频| 91试看福利一分钟| 中文字幕一区二区自拍| av中文字幕在线导航| 91精品高清一区二区三区| 亚洲一区自拍高清免费视频| 韩国一级特黄大片做受| 99视频精品全部15| 午夜福利人人妻人人澡人人爽| 黄色片黄色片wyaa| 秋霞午夜av福利经典影视| 亚洲在线一区二区欧美| 欧美精品一区二区三区xxxx| 亚洲国产精品久久久久蜜桃| 2020国产在线不卡视频| 亚洲av日韩av网站| 国产女人被做到高潮免费视频 | 激情内射在线免费观看| 视频一区二区综合精品| 日本xx片在线观看| 日本av在线一区二区三区| 成人av天堂丝袜在线观看| 国产精品免费不卡av| 午夜在线观看一区视频| 黑人大几巴狂插日本少妇| 夜夜操,天天操,狠狠操| 成人蜜桃美臀九一一区二区三区| 欧美日本aⅴ免费视频| 欧美另类一区二区视频| 偷拍自拍视频图片免费| 国产精品福利小视频a| 在线观看亚洲人成免费网址| 男人天堂色男人av| 亚洲国产欧美一区二区丝袜黑人| 欧美日韩国产一区二区三区三州| 人妻丝袜榨强中文字幕| 抽查舔水白紧大视频| 福利午夜视频在线合集| 青青青青青手机视频| 青青草原网站在线观看| 亚洲av无女神免非久久| 中文字幕在线一区精品| 日本人妻精品久久久久久| 亚洲av人人澡人人爽人人爱| 免费黄页网站4188| 日本熟妇喷水xxx| 日本午夜久久女同精女女| 久久久久久97三级| 91桃色成人网络在线观看| 精品日产卡一卡二卡国色天香| 97精品成人一区二区三区| 国产三级精品三级在线不卡| 视频一区二区在线免费播放| 天天干夜夜操天天舔| 欧美特级特黄a大片免费| 国产av国片精品一区二区| 欧美香蕉人妻精品一区二区| 在线成人日韩av电影| 激情五月婷婷综合色啪| 自拍偷拍 国产资源| 91she九色精品国产| 超pen在线观看视频公开97| 51国产偷自视频在线播放| 日日日日日日日日夜夜夜夜夜夜| 国产福利小视频二区| 亚洲精品无码色午夜福利理论片| 日韩剧情片电影在线收看| 久碰精品少妇中文字幕av| 日韩成人性色生活片| 国产精品久久久黄网站| 五十路息与子猛烈交尾视频| 久碰精品少妇中文字幕av| av大全在线播放免费| 亚洲一区二区三区久久午夜| 99热久久这里只有精品| 人人妻人人人操人人人爽| 521精品视频在线观看| 日韩美在线观看视频黄| 性感美女福利视频网站| 97色视频在线观看| 日本18禁久久久久久| 大学生A级毛片免费视频| 97精品视频在线观看| 在线免费91激情四射 | 国产女人被做到高潮免费视频 | 婷婷久久久久深爱网| yy96视频在线观看| 又大又湿又爽又紧A视频| 视频一区二区综合精品| 国产自拍在线观看成人| 日本熟妇喷水xxx| 国产三级片久久久久久久| 国产精品自拍在线视频| 美女张开两腿让男人桶av| 超污视频在线观看污污污| 青草久久视频在线观看| 精品黑人巨大在线一区| 中国黄色av一级片| 一区二区视频在线观看免费观看| 日本一区美女福利视频| 亚洲专区激情在线观看视频| 国产之丝袜脚在线一区二区三区| 欧美日韩不卡一区不区二区| 亚洲高清自偷揄拍自拍| gav成人免费播放| 亚洲国产最大av综合| 久久永久免费精品人妻专区| 国产实拍勾搭女技师av在线| 午夜美女少妇福利视频| 男人的天堂av日韩亚洲| 国产大学生援交正在播放| 男人的网址你懂的亚洲欧洲av | 超碰97人人做人人爱| 97国产精品97久久| 极品性荡少妇一区二区色欲| 亚洲国产中文字幕啊啊啊不行了 | 国产视频精品资源网站| japanese五十路熟女熟妇| 色婷婷六月亚洲综合香蕉| 国产黄网站在线观看播放| 欧美成人猛片aaaaaaa| 韩国三级aaaaa高清视频| 亚洲国际青青操综合网站| av天堂中文免费在线| 欧美在线偷拍视频免费看| 人人人妻人人澡人人| 玩弄人妻熟妇性色av少妇| av中文字幕国产在线观看| 亚洲av可乐操首页| 一级A一级a爰片免费免会员| 精产国品久久一二三产区区别 | 黑人解禁人妻叶爱071| 国产一区自拍黄视频免费观看| 午夜精品一区二区三区4| 在线免费观看亚洲精品电影 | 国产密臀av一区二区三| 99热这里只有国产精品6| 人妻少妇精品久久久久久| 亚洲av日韩高清hd| 日韩精品中文字幕播放| 国产大学生援交正在播放| 一区二区三区日韩久久| 国产亚洲四十路五十路| 亚洲中文字幕国产日韩| 欧美色婷婷综合在线| 懂色av蜜桃a v| jiuse91九色视频| 国产黄色片在线收看| 青青草原网站在线观看| 一区二区视频在线观看免费观看 | 午夜免费体验区在线观看| 亚洲另类在线免费观看| 男大肉棒猛烈插女免费视频 | 男人的天堂一区二区在线观看| 超碰公开大香蕉97| 色秀欧美视频第一页| 成人免费毛片aaaa| 亚洲精品久久视频婷婷| 久久久久久久精品老熟妇| 粉嫩小穴流水视频在线观看| 天天做天天干天天舔| 999九九久久久精品| 成人蜜桃美臀九一一区二区三区| 亚洲第一黄色在线观看| 国产高清女主播在线| 成人资源在线观看免费官网| 亚洲高清视频在线不卡| 97精品人妻一区二区三区精品| 黑人解禁人妻叶爱071| 成熟熟女国产精品一区| 亚洲激情偷拍一区二区| 人人爱人人妻人人澡39| 91极品大一女神正在播放| 亚洲av极品精品在线观看| 人妻久久久精品69系列| 欧美久久一区二区伊人| 日日操综合成人av| 亚洲伊人av天堂有码在线| 久久丁香花五月天色婷婷| 亚洲成人国产av在线| 色综合色综合色综合色| 亚洲精品午夜aaa久久| 久久尻中国美女视频| 欧美怡红院视频在线观看| 好男人视频在线免费观看网站| 在线免费观看欧美小视频| 亚洲成高清a人片在线观看| 高潮视频在线快速观看国家快速| 中文字幕一区二 区二三区四区| 韩国AV无码不卡在线播放| 80电影天堂网官网| 91精品激情五月婷婷在线| 一区二区三区四区视频在线播放| 日本最新一二三区不卡在线| 亚洲精品国品乱码久久久久| 欧美一区二区三区啪啪同性| 亚洲成人av一区在线| 91精品啪在线免费| 在线视频免费观看网| 精品91高清在线观看| 国产精品系列在线观看一区二区 | 久久久精品欧洲亚洲av| 天堂资源网av中文字幕| 91精品国产高清自在线看香蕉网| 粗大的内捧猛烈进出爽大牛汉子| 激情啪啪啪啪一区二区三区 | 狠狠操狠狠操免费视频| 首之国产AV医生和护士小芳| 九色精品视频在线播放| 午夜激情精品福利视频| 在线观看操大逼视频| 性感美女诱惑福利视频| 亚洲av色香蕉一区二区三区| 自拍偷拍日韩欧美一区二区| 久久久久久9999久久久久| 黄片大全在线观看观看| 久久久久久9999久久久久| 自拍偷区二区三区麻豆| 91国内视频在线观看| 久久精品36亚洲精品束缚| 亚洲福利午夜久久久精品电影网| 国产精品欧美日韩区二区| 在线播放一区二区三区Av无码| 亚洲国产第一页在线观看| 91精品国产观看免费| 亚洲精品国产在线电影| 在线制服丝袜中文字幕| 青青色国产视频在线| 欧美久久久久久三级网| 亚洲综合乱码一区二区| 精品人妻一二三区久久| 人妻丝袜av在线播放网址| 操操网操操伊剧情片中文字幕网| 欧美精品一区二区三区xxxx| 91久久综合男人天堂| 极品性荡少妇一区二区色欲| 国产性感美女福利视频| 偷青青国产精品青青在线观看| 欧美色婷婷综合在线| 欧美一级色视频美日韩| 午夜青青草原网在线观看| 国产使劲操在线播放| 久草视频在线免播放| 久久h视频在线观看| 黄色成人在线中文字幕| 青草亚洲视频在线观看| av高潮迭起在线观看| 日日爽天天干夜夜操| 国产久久久精品毛片| 亚洲 中文 自拍 无码| 最新日韩av传媒在线| 亚洲成高清a人片在线观看| 无码国产精品一区二区高潮久久4| 99一区二区在线观看| 国产又粗又黄又硬又爽| 成人乱码一区二区三区av| 免费岛国喷水视频在线观看| 一二三区在线观看视频| 黑人巨大的吊bdsm| 日本丰满熟妇BBXBBXHD| 国产露脸对白在线观看| 91成人精品亚洲国产| 亚洲图片偷拍自拍区| 亚洲综合一区成人在线| 人妻少妇av在线观看| 国产chinesehd精品麻豆| 久久久久只精品国产三级| 日韩a级精品一区二区| 久久久麻豆精亚洲av麻花| 九色精品视频在线播放| 国产亚洲精品视频合集| 国产高清在线在线视频| 国产女人露脸高潮对白视频| 久久99久久99精品影院| 果冻传媒av一区二区三区| 亚洲一区二区三区久久午夜| 日韩中文字幕福利av| 久久这里只有精品热视频| 黄色片黄色片wyaa| 精品一区二区三区欧美| 久久这里只有精品热视频| 91超碰青青中文字幕| 国产又色又刺激在线视频| 日本后入视频在线观看| 91九色porny国产蝌蚪视频| 日本高清撒尿pissing| 99国产精品窥熟女精品| 啊用力插好舒服视频| 日韩欧美中文国产在线| 日韩欧美一级精品在线观看| 青青青aaaa免费| 久久这里只有精彩视频免费| 国产精品人妻66p| 欧美偷拍亚洲一区二区| 午夜精品福利一区二区三区p| 五月精品丁香久久久久福利社| 日本女大学生的黄色小视频| 中文字幕熟女人妻久久久| 久碰精品少妇中文字幕av| 91精品国产91青青碰| 欧美黑人性猛交xxxxⅹooo| 日韩人妻在线视频免费| 久久久精品999精品日本| 亚洲午夜伦理视频在线| 青娱乐蜜桃臀av色| 在线免费观看亚洲精品电影| 成人蜜桃美臀九一一区二区三区 | 国产午夜激情福利小视频在线| 亚洲综合在线观看免费| 欧洲日韩亚洲一区二区三区| 91精品综合久久久久3d动漫| 男生用鸡操女生视频动漫| 视频在线免费观看你懂得| 男人操女人逼逼视频网站| av中文字幕国产在线观看| 亚洲狠狠婷婷综合久久app| 亚洲av色图18p| 国产夫妻视频在线观看免费| 国产日韩一区二区在线看| 午夜精品在线视频一区| 日韩精品激情在线观看| 热思思国产99re| 日韩加勒比东京热二区| 成人色综合中文字幕| av在线免费资源站| 亚洲成人av一区在线| 欧美天堂av无线av欧美| 国产+亚洲+欧美+另类| 青青青青青手机视频| 国产精品污污污久久| 国产日韩欧美视频在线导航| 中文亚洲欧美日韩无线码| 亚洲青青操骚货在线视频| 亚洲少妇人妻无码精品| 亚洲一区二区三区久久午夜| 操人妻嗷嗷叫视频一区二区| 国产一区二区神马久久| 成年人啪啪视频在线观看| 啊用力插好舒服视频| 免费在线观看视频啪啪| 青青社区2国产视频| 一区二区三区四区中文| 美女小视频网站在线| 国产欧美精品不卡在线| 成人H精品动漫在线无码播放| 超污视频在线观看污污污 | 国产精品黄大片在线播放| 天天操天天干天天日狠狠插| 久久h视频在线观看| 亚洲人成精品久久久久久久| 91久久国产成人免费网站| 亚洲天堂第一页中文字幕| 亚洲欧美综合另类13p| 国产精品sm调教视频| 午夜精品福利91av| 动漫av网站18禁| 国产视频网站一区二区三区| 啪啪啪操人视频在线播放| 97精品视频在线观看| 秋霞午夜av福利经典影视| 制丝袜业一区二区三区| 亚洲午夜高清在线观看| 人妻久久久精品69系列| 国产亚洲精品欧洲在线观看| 美女骚逼日出水来了| 9l人妻人人爽人人爽| 97精品综合久久在线| 青青尤物在线观看视频网站 | 欧美黄片精彩在线免费观看| 一区二区三区综合视频| 亚洲伊人av天堂有码在线| 精品一区二区三区三区88| 欧美精产国品一二三产品区别大吗| 日韩美av高清在线| 经典国语激情内射视频| 大鸡吧插逼逼视频免费看 | 日本一区精品视频在线观看| 一区二区三区四区中文| 黄色黄色黄片78在线| 老司机免费视频网站在线看| 国产精品久久久久久美女校花| 深田咏美亚洲一区二区| 老鸭窝日韩精品视频观看| 国产刺激激情美女网站| 精品欧美一区二区vr在线观看| 久久这里只有精彩视频免费| 51国产成人精品视频| 欧美日本国产自视大全| 欧美视频不卡一区四区| 在线观看免费视频色97| www骚国产精品视频| 巨乳人妻日下部加奈被邻居中出| 成人蜜桃美臀九一一区二区三区| 清纯美女在线观看国产| 色偷偷伊人大杳蕉综合网 | 插小穴高清无码中文字幕| 亚洲 国产 成人 在线| 青草亚洲视频在线观看| 国产亚洲四十路五十路| 亚洲av在线观看尤物| 啪啪啪啪啪啪啪啪av| 丰满少妇人妻xxxxx| 日本又色又爽又黄又粗| 日韩a级黄色小视频| 天美传媒mv视频在线观看| 新婚人妻聚会被中出| 夜夜嗨av一区二区三区中文字幕| 视频一区 视频二区 视频| 熟女俱乐部一二三区| 老司机免费视频网站在线看| 一区二区三区激情在线| 国产精品伦理片一区二区| 3344免费偷拍视频| 日韩av熟妇在线观看| 韩国AV无码不卡在线播放| 欧美区一区二区三视频| 啪啪啪啪啪啪啪啪av| av中文字幕国产在线观看| 亚洲激情唯美亚洲激情图片| 欧美精品激情在线最新观看视频 | 欧美日韩在线精品一区二区三| 久久精品亚洲成在人线a| 亚洲成a人片777777| 日本韩国亚洲综合日韩欧美国产| 麻豆精品成人免费视频| 欧美在线精品一区二区三区视频| 欧美aa一级一区三区四区| 亚洲国产成人av在线一区| 天天插天天狠天天操| 美女av色播在线播放| 欧美精品亚洲精品日韩在线| 欧美交性又色又爽又黄麻豆| 北条麻妃肉色丝袜视频| 2017亚洲男人天堂| 中文字幕乱码人妻电影| 午夜精品一区二区三区福利视频| 亚洲的电影一区二区三区| 色av色婷婷人妻久久久精品高清| 日本人妻少妇18—xx| 一级A一级a爰片免费免会员| 国产黄色片在线收看| 麻豆性色视频在线观看| 成人国产激情自拍三区| 亚洲国产成人最新资源| 欧洲精品第一页欧洲精品亚洲| 最后99天全集在线观看| 成人av久久精品一区二区| 国产使劲操在线播放| av在线免费中文字幕| 色哟哟国产精品入口| 偷拍自拍 中文字幕| 视频啪啪啪免费观看| 中文字幕日韩精品就在这里| 日韩在线视频观看有码在线| 亚洲国产40页第21页| asmr福利视频在线观看| 国产综合视频在线看片| 夜色17s精品人妻熟女| 深夜男人福利在线观看| 天天操夜夜骑日日摸| 五月天色婷婷在线观看视频免费| 超黄超污网站在线观看| 亚洲精品精品国产综合| 亚洲区欧美区另类最新章节| 一区二区三区麻豆福利视频| 亚洲精品麻豆免费在线观看| 中文字幕视频一区二区在线观看 | 中文字幕国产专区欧美激情| weyvv5国产成人精品的视频| 天天色天天爱天天爽| 精品一区二区亚洲欧美| 男女啪啪视频免费在线观看 | 国产精品视频男人的天堂| 天天日天天透天天操| 色综合久久久久久久久中文| 91免费福利网91麻豆国产精品| 黄色中文字幕在线播放| 99的爱精品免费视频| 端庄人妻堕落挣扎沉沦| 日韩av有码中文字幕| 偷青青国产精品青青在线观看| 日韩美女搞黄视频免费| 日本熟妇一区二区x x| 日本人妻欲求不满中文字幕| 特黄老太婆aa毛毛片| 天干天天天色天天日天天射| 桃色视频在线观看一区二区| 久久这里只有精品热视频| 黄工厂精品视频在线观看 | 成人福利视频免费在线| 在线观看av观看av| 国产卡一卡二卡三乱码手机| 日本一本午夜在线播放| 国产美女一区在线观看| 国产精品视频男人的天堂| 亚洲av天堂在线播放| 国产露脸对白在线观看| 亚洲伊人色一综合网| 国产精品免费不卡av| 午夜激情精品福利视频| 欧美va不卡视频在线观看 | 91精品高清一区二区三区| 国产成人无码精品久久久电影| 热思思国产99re| 国产精品自拍偷拍a| 精品亚洲国产中文自在线| 日本欧美视频在线观看三区| 日本性感美女三级视频| 青青青青爽手机在线| 日本午夜久久女同精女女| 啪啪啪18禁一区二区三区| 国产美女精品福利在线| 操日韩美女视频在线免费看| 亚洲av琪琪男人的天堂| 999九九久久久精品| yy96视频在线观看| 天天日天天玩天天摸| av森泽佳奈在线观看| 在线观看av观看av| 亚洲一区二区三区五区 | 亚洲av第国产精品| 青青青视频自偷自拍38碰| 国产精品午夜国产小视频| 精品亚洲在线免费观看| 欧美亚洲中文字幕一区二区三区| 亚洲av无硬久久精品蜜桃| 日日爽天天干夜夜操| 国产丰满熟女成人视频| 老鸭窝日韩精品视频观看| 偷拍自拍国产在线视频| 中文字幕av一区在线观看| 国产综合高清在线观看| 亚洲精品福利网站图片| 欧洲日韩亚洲一区二区三区| 精品91高清在线观看 | 国产黄网站在线观看播放| 午夜精彩视频免费一区| 国产一区二区火爆视频 | 亚洲熟色妇av日韩熟色妇在线| 中文字幕 人妻精品| 99热99这里精品6国产| 久久永久免费精品人妻专区| 日本韩国在线观看一区二区| 非洲黑人一级特黄片| 亚洲av琪琪男人的天堂| 国产av一区2区3区| 欧美日韩在线精品一区二区三| 日日夜夜狠狠干视频| 久久久久久久一区二区三| 久久丁香婷婷六月天| 美女骚逼日出水来了| 91国偷自产一区二区三区精品| 懂色av之国产精品| 天天艹天天干天天操| 天堂av在线最新版在线| 日本一二三区不卡无| 五月天久久激情视频| 欧美色婷婷综合在线| 亚洲中文字幕校园春色| 青青尤物在线观看视频网站| 自拍偷拍一区二区三区图片| 99视频精品全部15| 日本三极片视频网站观看| 福利国产视频在线观看| 综合国产成人在线观看| 亚洲精品三级av在线免费观看| 午夜的视频在线观看| 日韩人妻丝袜中文字幕| 亚洲伊人久久精品影院一美女洗澡| 日韩精品啪啪视频一道免费| 伊人成人综合开心网| 亚洲一区二区三区久久受| 五十路丰满人妻熟妇| 男生用鸡操女生视频动漫| 在线视频国产欧美日韩| wwwxxx一级黄色片| 粉嫩av蜜乳av蜜臀| 91福利在线视频免费观看| 天天色天天舔天天射天天爽| 婷婷久久久综合中文字幕| 欧美麻豆av在线播放| 五色婷婷综合狠狠爱| 国产熟妇乱妇熟色T区| 五色婷婷综合狠狠爱| 日韩少妇人妻精品无码专区| 99re久久这里都是精品视频| 91精品国产高清自在线看香蕉网| 超碰在线观看免费在线观看| 中文亚洲欧美日韩无线码| 777奇米久久精品一区| 午夜久久香蕉电影网| 久久尻中国美女视频| 夜夜嗨av蜜臀av| 91天堂天天日天天操| 久久热久久视频在线观看| 亚洲第一伊人天堂网| 51国产成人精品视频 | 中国把吊插入阴蒂的视频| 一区二区三区四区中文| 亚洲激情唯美亚洲激情图片| 91九色国产porny蝌蚪| 九色porny九色9l自拍视频| 亚洲欧洲一区二区在线观看| 天天日天天天天天天天天天天| 亚洲蜜臀av一区二区三区九色 | 非洲黑人一级特黄片| 亚洲成人激情av在线| 国产亚洲视频在线二区| 都市家庭人妻激情自拍视频| 超级碰碰在线视频免费观看| 日韩国产乱码中文字幕| 人妻无码色噜噜狠狠狠狠色| 国产欧美日韩在线观看不卡| 午夜青青草原网在线观看| 99精品视频之69精品视频 | 班长撕开乳罩揉我胸好爽| 精品区一区二区三区四区人妻| av久久精品北条麻妃av观看| 在线免费观看av日韩| 亚洲青青操骚货在线视频| 日本a级视频老女人| 天天夜天天日天天日| 91国产在线免费播放| 亚洲国产精品中文字幕网站| 婷婷久久久综合中文字幕| 免费在线播放a级片| 日本精品美女在线观看| 丰满的子国产在线观看| 93视频一区二区三区| 久久久久久久久久久久久97| 2020av天堂网在线观看| 成人亚洲国产综合精品| 97超碰免费在线视频| 亚洲av色图18p| 中国熟女@视频91| 九色精品视频在线播放| 4个黑人操素人视频网站精品91| 日韩美女综合中文字幕pp| 社区自拍揄拍尻屁你懂的| 日韩精品中文字幕在线| 欧美精产国品一二三产品价格| 亚洲免费国产在线日韩| 色哟哟在线网站入口| 99热久久极品热亚洲| 免费国产性生活视频| 福利视频广场一区二区| 老司机免费视频网站在线看| 亚洲熟女女同志女同| 亚洲欧美激情人妻偷拍| 无码精品一区二区三区人| 精品一区二区三区欧美| 亚洲成人国产av在线| 中文字幕亚洲久久久| 国产熟妇一区二区三区av| 少妇露脸深喉口爆吞精| 18禁免费av网站| 国产精品人妻熟女毛片av久| rct470中文字幕在线| 亚洲 色图 偷拍 欧美| 家庭女教师中文字幕在线播放| 99国内小视频在现欢看| 久草电影免费在线观看| 日日夜夜精品一二三| 精品日产卡一卡二卡国色天香 | www天堂在线久久| 天天日天天干天天插舔舔| 久久机热/这里只有| 1区2区3区4区视频在线观看| 国产精品国产精品一区二区| 视频一区 二区 三区 综合| 亚洲卡1卡2卡三卡四老狼| 日本阿v视频在线免费观看| 岛国av高清在线成人在线| 中文字幕免费福利视频6| 国产黄色大片在线免费播放| 天天操天天污天天射| 2018最新中文字幕在线观看| 欧美老鸡巴日小嫩逼| 国产高清在线观看1区2区| 日韩av熟妇在线观看| 2021最新热播中文字幕| 亚洲精品久久综合久| 国产真实乱子伦a视频| 亚洲午夜在线视频福利| 岳太深了紧紧的中文字幕| 色噜噜噜噜18禁止观看| 国产在线拍揄自揄视频网站| 在线观看免费视频网| 韩国爱爱视频中文字幕| 女警官打开双腿沦为性奴| 中文字幕av一区在线观看| 在线观看av亚洲情色| 国产亚州色婷婷久久99精品| 男人天堂色男人av| 一区二区久久成人网| 国产精品久久久久国产三级试频| 亚洲一级av无码一级久久精品| 国产av一区2区3区| 国产乱子伦精品视频潮优女| 亚洲欧美综合在线探花| 99久久99久国产黄毛片| 91国内精品自线在拍白富美| 日韩中文字幕在线播放第二页| 少妇高潮一区二区三区| 成人av免费不卡在线观看| 日噜噜噜夜夜噜噜噜天天噜噜噜| 偷拍自拍亚洲美腿丝袜| 97成人免费在线观看网站| 国产午夜亚洲精品不卡在线观看| 精品人妻伦一二三区久| 99精品免费观看视频| 性感美女高潮视频久久久 | 国产乱子伦一二三区| 欧美视频一区免费在线| 国产午夜激情福利小视频在线| 真实国模和老外性视频| 欧美亚洲偷拍自拍色图| 毛片一级完整版免费| 一级黄片大鸡巴插入美女| 三级等保密码要求条款| 大香蕉伊人国产在线| 888亚洲欧美国产va在线播放| 欧美viboss性丰满| 日噜噜噜夜夜噜噜噜天天噜噜噜| 日本熟妇丰满厨房55| 日本一区二区三区免费小视频| 一区二区三区四区视频| 中文字幕AV在线免费看 | 一区国内二区日韩三区欧美| 人人妻人人爽人人添夜| 97超碰最新免费在线观看| 亚洲av色香蕉一区二区三区| 国产成人精品福利短视频| 少妇露脸深喉口爆吞精| 老司机99精品视频在线观看 | 国产剧情演绎系列丝袜高跟| 黄色视频在线观看高清无码| 一区二区视频视频视频| 最新黄色av网站在线观看| 日日操夜夜撸天天干| 人妻素人精油按摩中出| 少妇深喉口爆吞精韩国| 在线观看的黄色免费网站| 久久丁香婷婷六月天| 国产揄拍高清国内精品对白| 黄工厂精品视频在线观看| 91国语爽死我了不卡| 青青青青草手机在线视频免费看| 经典国语激情内射视频| 日韩欧美国产精品91| av高潮迭起在线观看| 一区二区在线视频中文字幕| 亚洲精品一线二线在线观看| 欧美黄片精彩在线免费观看| av天堂中文免费在线| 精品91自产拍在线观看一区| 99re久久这里都是精品视频| 激情伦理欧美日韩中文字幕| 日本www中文字幕| 又色又爽又黄的美女裸体| 亚洲av无码成人精品区辽| 国产精品人妻一区二区三区网站 | 色婷婷六月亚洲综合香蕉| 中文字幕熟女人妻久久久| 人人妻人人澡欧美91精品| 91大神福利视频网| 中文字幕午夜免费福利视频| 亚洲1卡2卡三卡4卡在线观看| 中文字幕日韩精品日本| 天天综合天天综合天天网| 国产福利小视频二区| 午夜美女少妇福利视频| 天天躁日日躁狠狠躁av麻豆| 青青草亚洲国产精品视频| www日韩a级s片av| 久久久制服丝袜中文字幕| 亚洲精品在线资源站| 亚洲av成人免费网站| 毛片一级完整版免费| 中文字幕AV在线免费看 | 老有所依在线观看完整版| 唐人色亚洲av嫩草| 大鸡巴操b视频在线| 国产综合高清在线观看| 日辽宁老肥女在线观看视频| 男大肉棒猛烈插女免费视频| 欧美综合婷婷欧美综合| 中文字幕一区二区人妻电影冢本| av无限看熟女人妻另类av| 一区二区三区综合视频| 北条麻妃肉色丝袜视频| 沙月文乃人妻侵犯中文字幕在线| 国产精品入口麻豆啊啊啊| 少妇一区二区三区久久久| 日韩中文字幕精品淫| 成人免费公开视频无毒| 日韩精品激情在线观看| av在线播放国产不卡| 日韩欧美国产一区ab| 久久三久久三久久三久久| 日本韩国免费一区二区三区视频| 亚洲人人妻一区二区三区 | 国产午夜无码福利在线看| 毛片av在线免费看| 人人妻人人人操人人人爽| 啪啪啪啪啪啪啪免费视频| 91精品国产综合久久久蜜 | 青青伊人一精品视频| AV无码一区二区三区不卡| 97超碰最新免费在线观看| 久久免看30视频口爆视频| 亚洲国产成人无码麻豆艾秋| 91国产在线视频免费观看| 美女张开腿让男生操在线看| 婷婷综合亚洲爱久久| 3344免费偷拍视频| 日韩熟女系列一区二区三区| 精品国产污污免费网站入口自| 91大神福利视频网| 蜜臀av久久久久久久| 清纯美女在线观看国产| 粉嫩欧美美人妻小视频| 中文字幕成人日韩欧美| 美洲精品一二三产区区别| 欧美80老妇人性视频| 青青青青青青青在线播放视频| 在线观看黄色成年人网站| 中国黄片视频一区91| 特黄老太婆aa毛毛片| 五十路熟女人妻一区二| 天天日夜夜操天天摸| 欧美日韩在线精品一区二区三| 2020国产在线不卡视频| v888av在线观看视频| 天天插天天色天天日| 欧洲国产成人精品91铁牛tv| 亚洲熟女女同志女同| 免费一级特黄特色大片在线观看| 51国产成人精品视频| 日韩三级黄色片网站| 精品一线二线三线日本| 在线免费观看欧美小视频| 2017亚洲男人天堂| 欧美成人一二三在线网| 亚洲成人三级在线播放| 日本免费午夜视频网站| 中文字幕一区的人妻欧美日韩| 精品久久久久久久久久久a√国产| 午夜精品福利91av| 青青热久免费精品视频在线观看| 天天做天天干天天舔| 97香蕉碰碰人妻国产樱花| 亚洲区欧美区另类最新章节| 老鸭窝在线观看一区| 日本中文字幕一二区视频| 天天色天天操天天舔| 91精品资源免费观看| 亚洲国产40页第21页| 激情内射在线免费观看| 好吊视频—区二区三区| 久久久久久久精品成人热| 乱亲女秽乱长久久久| 日本高清撒尿pissing| 九九视频在线精品播放| 老司机福利精品视频在线| 久久香蕉国产免费天天| 扒开让我视频在线观看| 51国产偷自视频在线播放| 亚洲 中文 自拍 另类 欧美| 啊慢点鸡巴太大了啊舒服视频| 1769国产精品视频免费观看| 青青草成人福利电影| 国产精品系列在线观看一区二区| 2022国产综合在线干| 中文字幕av男人天堂| 国产熟妇乱妇熟色T区| 视频在线免费观看你懂得| 91 亚洲视频在线观看| 自拍 日韩 欧美激情| 免费人成黄页网站在线观看国产 | 久久尻中国美女视频| 国产一区av澳门在线观看| 高清一区二区欧美系列| 韩国黄色一级二级三级| 日韩a级黄色小视频| 亚洲综合另类欧美久久| 天堂av在线播放免费| 成人蜜桃美臀九一一区二区三区| 天堂av在线播放免费| 亚洲av无码成人精品区辽| 亚洲熟女久久久36d| 韩国黄色一级二级三级| 超级福利视频在线观看| 日本av在线一区二区三区| 亚洲男人让女人爽的视频| 一级黄色片夫妻性生活| 91试看福利一分钟| 无码日韩人妻精品久久| 日本熟妇一区二区x x| 999热精品视频在线| 亚洲成人av一区在线| 亚洲最大黄了色网站| 91人妻精品久久久久久久网站 | 性感美女诱惑福利视频| 国产精彩福利精品视频| 午夜精品一区二区三区更新| 国产午夜福利av导航| 鸡巴操逼一级黄色气| 亚洲av男人天堂久久| 国产揄拍高清国内精品对白| 亚洲午夜伦理视频在线| 国产V亚洲V天堂无码欠欠| AV无码一区二区三区不卡| 日韩一个色综合导航| 天天摸天天干天天操科普| 午夜大尺度无码福利视频| 亚洲av色图18p| 成人性黑人一级av| 亚洲精品成人网久久久久久小说| 日本乱人一区二区三区| 天堂av在线官网中文| 精品久久久久久久久久久久人妻| 精品av国产一区二区三区四区 | 超碰在线观看免费在线观看| 亚洲色偷偷综合亚洲AV伊人| 国产成人一区二区三区电影网站| 在线观看免费岛国av| 午夜精品一区二区三区城中村| 日韩熟女系列一区二区三区| 日本少妇在线视频大香蕉在线观看 | 福利午夜视频在线观看| 四川五十路熟女av| 狍和女人的王色毛片| 国产精品免费不卡av| 一区二区视频视频视频| 国产久久久精品毛片| 婷婷久久久综合中文字幕| 精品av国产一区二区三区四区 | 国产午夜无码福利在线看| 日本一二三中文字幕| 丝袜肉丝一区二区三区四区在线| 成人av亚洲一区二区| 成年女人免费播放视频| 一区二区久久成人网| 久草视频在线免播放| 国产三级精品三级在线不卡| 国产在线观看免费人成短视频| 人妻最新视频在线免费观看| 91国内精品自线在拍白富美| 免费岛国喷水视频在线观看| 97人妻人人澡爽人人精品| 亚洲中文字幕校园春色| 六月婷婷激情一区二区三区| 国产精品久久久久久久女人18| sejizz在线视频| 一区二区三区美女毛片| 岛国青草视频在线观看| 日韩欧美国产一区ab| 成人乱码一区二区三区av| 神马午夜在线观看视频| 色偷偷伊人大杳蕉综合网| 男人的天堂在线黄色| 99视频精品全部15| 2017亚洲男人天堂| 超鹏97历史在线观看| 五十路熟女av天堂| 成人免费公开视频无毒| 亚洲中文字幕人妻一区| 亚洲码av无色中文| 亚洲精品高清自拍av | 最新91精品视频在线 | 色狠狠av线不卡香蕉一区二区| 日本脱亚入欧是指什么| 97超碰最新免费在线观看| av完全免费在线观看av| 超级碰碰在线视频免费观看| 91色秘乱一区二区三区| 99精品国产自在现线观看| 婷婷色中文亚洲网68| 99热色原网这里只有精品| 天天操夜夜操天天操天天操| 动漫av网站18禁| 人妻丝袜精品中文字幕| 在线新三级黄伊人网| caoporm超碰国产| a v欧美一区=区三区| 人人妻人人爽人人添夜| 专门看国产熟妇的网站| 蜜臀av久久久久蜜臀av麻豆| 男人操女人逼逼视频网站| 韩国爱爱视频中文字幕| 国产日韩av一区二区在线| 女生自摸在线观看一区二区三区| 五月婷婷在线观看视频免费| 综合色区亚洲熟妇shxstz| 国产亚洲精品视频合集| 制丝袜业一区二区三区| 97成人免费在线观看网站| 国产综合高清在线观看| 国产女孩喷水在线观看| 专门看国产熟妇的网站| 熟女人妻一区二区精品视频| 大鸡吧插入女阴道黄色片| 亚洲最大免费在线观看| 久久久精品精品视频视频| 丝袜美腿欧美另类 中文字幕| 偷拍自拍亚洲视频在线观看| 五十路av熟女松本翔子| 久草极品美女视频在线观看| 亚洲图片偷拍自拍区| 性色av一区二区三区久久久| 91福利在线视频免费观看| 懂色av之国产精品| 班长撕开乳罩揉我胸好爽| 少妇与子乱在线观看| 午夜精品久久久久麻豆影视| 18禁精品网站久久| 亚洲欧洲一区二区在线观看| 日韩熟女av天堂系列| 熟女视频一区,二区,三区| 在线观看视频一区麻豆| 亚洲视频在线观看高清| 男女之间激情网午夜在线| 亚洲福利天堂久久久久久| 亚洲欧美人精品高清| 国产午夜亚洲精品不卡在线观看| 久久久久久cao我的性感人妻| 少妇人妻100系列| 亚洲av自拍偷拍综合| 亚洲国产欧美一区二区三区久久| 久草免费人妻视频在线| 老司机欧美视频在线看| 成人资源在线观看免费官网| 东游记中文字幕版哪里可以看到| 被大鸡吧操的好舒服视频免费| huangse网站在线观看| 亚洲精品三级av在线免费观看| 中文字幕乱码av资源| 国产精品大陆在线2019不卡| 精品91高清在线观看| 亚洲最大黄了色网站| 97人妻夜夜爽二区欧美极品| 在线观看免费岛国av| 亚洲成人av一区久久| 天天综合天天综合天天网| 真实国产乱子伦一区二区| 日韩一区二区三区三州| v888av在线观看视频| 中文字幕综合一区二区| 一区二区三区四区视频在线播放| 国产男女视频在线播放| 天堂av狠狠操蜜桃| 日韩欧美在线观看不卡一区二区| 精品美女久久久久久| 国产超码片内射在线| 国产内射中出在线观看| 521精品视频在线观看| 婷婷综合蜜桃av在线| 成人精品在线观看视频| 天天操天天插天天色| 日韩在线视频观看有码在线| 美女日逼视频免费观看| 午夜毛片不卡在线看| 五十路熟女av天堂| 在线观看911精品国产| www日韩a级s片av| 日日夜夜大香蕉伊人| 欧美色呦呦最新网址| 福利片区一区二体验区| 人妻久久无码中文成人| 亚洲免费成人a v| 少妇高潮无套内谢麻豆| 东游记中文字幕版哪里可以看到| 老司机福利精品视频在线| 国产视频一区在线观看| 经典av尤物一区二区| 黄色无码鸡吧操逼视频| gav成人免费播放| 成年人免费看在线视频| 中文字幕之无码色多多| 91桃色成人网络在线观看| AV无码一区二区三区不卡| 精品久久久久久久久久久99| 老司机福利精品视频在线| 熟妇一区二区三区高清版| 动漫av网站18禁| 男人在床上插女人视频| 激情色图一区二区三区| 天天干天天爱天天色| 热思思国产99re| 少妇高潮一区二区三区| 99精品国产自在现线观看| 亚洲av成人免费网站| 亚洲第17页国产精品| 狍和女人的王色毛片| 91精品国产综合久久久蜜| 熟女视频一区,二区,三区| 国产精品伦理片一区二区| 一本久久精品一区二区| 久久久人妻一区二区| 亚洲免费视频欧洲免费视频| 动漫美女的小穴视频| 色花堂在线av中文字幕九九| 2020国产在线不卡视频| 九色精品视频在线播放| 99热久久这里只有精品8| 国内精品在线播放第一页| 天天色天天操天天透| 久久久久91精品推荐99| 亚洲国际青青操综合网站| 中国熟女一区二区性xx| 999久久久久999| 果冻传媒av一区二区三区| 国产清纯美女al在线| 91精品国产麻豆国产| 97人人妻人人澡人人爽人人精品| 最新91九色国产在线观看| 视频二区在线视频观看| 国产精品久久综合久久| 97超碰人人搞人人| 激情人妻校园春色亚洲欧美 | 啊慢点鸡巴太大了啊舒服视频| 免费在线看的黄网站| 日本熟妇喷水xxx| 国产精品大陆在线2019不卡| 888欧美视频在线| 在线免费观看日本片| 国产精品视频一区在线播放| 国产精品一二三不卡带免费视频| 中文字幕av第1页中文字幕| 超碰97免费人妻麻豆| 日本性感美女视频网站| 蜜桃专区一区二区在线观看| 中文字幕人妻熟女在线电影| 夜女神免费福利视频| 一区二区麻豆传媒黄片| 色婷婷六月亚洲综合香蕉| 久久久久国产成人精品亚洲午夜| 天天干天天操天天玩天天射| 日韩av中文在线免费观看| 日本真人性生活视频免费看| 成人H精品动漫在线无码播放| 日韩av熟妇在线观看| 国产精品女邻居小骚货| av中文字幕网址在线| 沈阳熟妇28厘米大战黑人| 亚洲1069综合男同| 成人av免费不卡在线观看| 久久久久久久久久久久久97| 国产视频一区在线观看| 91精品免费久久久久久| 大香蕉福利在线观看| 美女日逼视频免费观看| 激情色图一区二区三区| 亚洲福利天堂久久久久久 | 超黄超污网站在线观看| 2020中文字幕在线播放| 国产在线自在拍91国语自产精品| 久久永久免费精品人妻专区| 精品久久久久久久久久久a√国产| 在线国产精品一区二区三区| av中文字幕电影在线看| 精品一线二线三线日本| 人妻少妇亚洲精品中文字幕| 狠狠躁狠狠爱网站视频| 少妇高潮一区二区三区| 18禁美女黄网站色大片下载| 大鸡巴操娇小玲珑的女孩逼| 亚洲av香蕉一区区二区三区犇| 午夜蜜桃一区二区三区| 蜜臀av久久久久蜜臀av麻豆| 欧美精产国品一二三产品价格| 1区2区3区不卡视频| 亚洲熟色妇av日韩熟色妇在线| 久久精品亚洲成在人线a| 日本少妇人妻xxxxx18| av欧美网站在线观看| 又黄又刺激的午夜小视频| 欧美精产国品一二三产品价格| 久久免费看少妇高潮完整版| 欧美aa一级一区三区四区| 国产中文精品在线观看| 岛国免费大片在线观看 | 日本性感美女写真视频| 北条麻妃高跟丝袜啪啪| 日韩中文字幕在线播放第二页| 92福利视频午夜1000看 | 国产精品污污污久久| 亚洲中文字幕人妻一区| 51国产偷自视频在线播放| 天天干天天搞天天摸| 午夜精品亚洲精品五月色| 欧美va亚洲va天堂va| 国产av国片精品一区二区| 色婷婷精品大在线观看| 丝袜肉丝一区二区三区四区在线| 亚洲男人让女人爽的视频| 人妻在线精品录音叫床| 成年人中文字幕在线观看| 国产极品精品免费视频| 宅男噜噜噜666国产| 大黑人性xxxxbbbb| 国产一级麻豆精品免费| 日本人妻欲求不满中文字幕| 欧美爆乳肉感大码在线观看| 中文字幕在线免费第一页| 免费看美女脱光衣服的视频| 成人精品视频99第一页| 久久久久久97三级| 麻豆性色视频在线观看| 中文字日产幕乱六区蜜桃 | avjpm亚洲伊人久久| 青青青青草手机在线视频免费看| 亚洲黄色av网站免费播放| 精品视频中文字幕在线播放| 日韩视频一区二区免费观看| 91色秘乱一区二区三区| 又粗又硬又猛又爽又黄的| 亚洲图片欧美校园春色| 成年女人免费播放视频| 国产午夜无码福利在线看| 国产高清在线观看1区2区| 视频一区二区在线免费播放| 欧美精品资源在线观看| 国产福利在线视频一区| 日韩av中文在线免费观看| 免费成人va在线观看| 美女大bxxxx内射| 啪啪啪啪啪啪啪啪av| 特级无码毛片免费视频播放| 韩国女主播精品视频网站| 国产精品系列在线观看一区二区| 三级黄色亚洲成人av| 国产成人精品午夜福利训2021| 在线免费视频 自拍| 国产精品熟女久久久久浪潮| 成人网18免费视频版国产| 中文字幕一区二区自拍| 日本在线一区二区不卡视频| 这里有精品成人国产99| 青青擦在线视频国产在线| 国产精品视频一区在线播放| 真实国模和老外性视频| 中文字幕亚洲中文字幕| 青青在线视频性感少妇和隔壁黑丝| 欧亚乱色一区二区三区| 国产福利小视频二区| 一区二区三区蜜臀在线| 天天日天天操天天摸天天舔| 欧美精品久久久久久影院| 久久久久久性虐视频| 中文字幕一区二 区二三区四区 | av手机免费在线观看高潮| 国产欧美日韩第三页| 青青操免费日综合视频观看| 四虎永久在线精品免费区二区| 人妻在线精品录音叫床| 黑人大几巴狂插日本少妇| 国产在线自在拍91国语自产精品| 中文字幕无码日韩专区免费| 免费看高清av的网站| 国产成人自拍视频播放| 国产精品一区二区久久久av| 亚洲av无硬久久精品蜜桃| 国产一区二区三免费视频| 精产国品久久一二三产区区别| 天天色天天爱天天爽| 精品成人午夜免费看| 99热色原网这里只有精品| 97人妻人人澡爽人人精品| 亚洲视频乱码在线观看| 熟女人妻在线中出观看完整版| 最新国产亚洲精品中文在线| 国产久久久精品毛片| 中国黄色av一级片| 天天做天天干天天舔| 国产a级毛久久久久精品| 40道精品招牌菜特色| 首之国产AV医生和护士小芳| 性色蜜臀av一区二区三区| 亚洲少妇人妻无码精品| 色综合色综合色综合色| 91精品国产综合久久久蜜| 中文字幕熟女人妻久久久| 精品一区二区三区在线观看| 欧美精产国品一二三产品区别大吗| av久久精品北条麻妃av观看| 亚洲一区二区三区av网站| xxx日本hd高清| 熟妇一区二区三区高清版| 伊人情人综合成人久久网小说| 日韩伦理短片在线观看| 青青青青草手机在线视频免费看| 在线视频免费观看网| 日本熟妇一区二区x x| 99久久超碰人妻国产| 国产午夜亚洲精品麻豆| 人人爽亚洲av人人爽av| 99热99这里精品6国产| 亚洲综合另类欧美久久| 夏目彩春在线中文字幕| 亚洲中文字幕校园春色| 日本脱亚入欧是指什么| 在线观看av2025| 国产精品成人xxxx| 天天日天天爽天天爽| 大陆精品一区二区三区久久| 瑟瑟视频在线观看免费视频| 精品黑人一区二区三区久久国产| 青青青视频手机在线观看| www日韩毛片av| 2022国产综合在线干| 最近的中文字幕在线mv视频| 日韩欧美制服诱惑一区在线| 制丝袜业一区二区三区| 久久久久久久久久久免费女人| 亚洲免费福利一区二区三区| 欧美交性又色又爽又黄麻豆| wwwxxx一级黄色片| 亚洲 中文 自拍 无码| 51国产成人精品视频| 中出中文字幕在线观看 | 国产黑丝高跟鞋视频在线播放| 男大肉棒猛烈插女免费视频 | av男人天堂狠狠干| 欧美一区二区三区高清不卡tv | 这里只有精品双飞在线播放| 极品性荡少妇一区二区色欲| 亚洲成人av一区久久| 天天干天天操天天爽天天摸 | 97精品成人一区二区三区 | 亚洲一区av中文字幕在线观看| 人妻少妇性色欲欧美日韩 | 天天想要天天操天天干| 91色秘乱一区二区三区| 免费观看理论片完整版| 精品人人人妻人人玩日产欧| 人妻少妇一区二区三区蜜桃| 天堂v男人视频在线观看| 亚洲卡1卡2卡三卡四老狼| 在线观看免费视频色97| 中文字幕第三十八页久久| 91精品国产91久久自产久强| 成人av免费不卡在线观看| 国产成人精品一区在线观看| 欧美一级片免费在线成人观看| 中文亚洲欧美日韩无线码| 在线播放国产黄色av| 91国偷自产一区二区三区精品| 久久农村老妇乱69系列| 久久久人妻一区二区| 日本少妇的秘密免费视频| 2018最新中文字幕在线观看| 黄工厂精品视频在线观看| 五月激情婷婷久久综合网| 国产黄色片蝌蚪九色91| 亚洲男人在线天堂网| 国产清纯美女al在线| 在线播放 日韩 av| 天天做天天干天天舔| 日韩a级黄色小视频| 班长撕开乳罩揉我胸好爽| 久久人人做人人妻人人玩精品vr| 一区二区三区综合视频| 97成人免费在线观看网站| 超碰在线观看免费在线观看| 国产va在线观看精品| 老司机深夜免费福利视频在线观看| 美女被肏内射视频网站| 日本一二三区不卡无| 欧美地区一二三专区| mm131美女午夜爽爽爽| 中文字幕一区二区三区人妻大片 | 激情综合治理六月婷婷| 看一级特黄a大片日本片黑人| 人妻最新视频在线免费观看| av老司机亚洲一区二区| 激情国产小视频在线| 国产又粗又猛又爽又黄的视频在线| 三级av中文字幕在线观看| 黄色黄色黄片78在线| caoporn蜜桃视频| 在线 中文字幕 一区| 人妻av无码专区久久绿巨人| 93人妻人人揉人人澡人人| av成人在线观看一区| 亚洲中文精品字幕在线观看| 中文字幕中文字幕 亚洲国产| 91麻豆精品久久久久| 偷拍自拍亚洲美腿丝袜| 成年美女黄网站18禁久久| 日韩美在线观看视频黄| 国产一区成人在线观看视频| 人妻少妇亚洲精品中文字幕| 精品乱子伦一区二区三区免费播| 天堂v男人视频在线观看| 大胆亚洲av日韩av| 成年人免费看在线视频| 欧洲精品第一页欧洲精品亚洲| 好吊操视频这里只有精品| 国产 在线 免费 精品| 夏目彩春在线中文字幕| 午夜美女少妇福利视频| 亚洲一区久久免费视频| 精品av国产一区二区三区四区 | 1区2区3区不卡视频| 成人精品视频99第一页| 直接能看的国产av| 91在线视频在线精品3| 中文字幕人妻三级在线观看 | 亚洲 清纯 国产com| 91九色国产熟女一区二区| 国产自拍在线观看成人| 老司机深夜免费福利视频在线观看| 成人精品在线观看视频| 3344免费偷拍视频| 天码人妻一区二区三区在线看| 91国内精品久久久久精品一| 日本韩国免费一区二区三区视频| 婷婷久久久综合中文字幕| 欧美另类重口味极品在线观看| 日韩熟女系列一区二区三区| 国产91久久精品一区二区字幕| 亚洲综合另类精品小说| 88成人免费av网站| 专门看国产熟妇的网站| 精品视频一区二区三区四区五区| 国产在线免费观看成人| 日本美女成人在线视频| 国产麻豆91在线视频| 摧残蹂躏av一二三区| 2020韩国午夜女主播在线| 99精品视频在线观看婷婷| 亚洲图片偷拍自拍区| 丝袜肉丝一区二区三区四区在线 | 日本a级视频老女人| 91九色porny国产蝌蚪视频| 亚洲av色图18p| 九九热99视频在线观看97| 中英文字幕av一区| 亚洲成人熟妇一区二区三区 | 日本一本午夜在线播放| 亚洲欧美久久久久久久久|