精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

后端思維之通過層層代碼去重,我又搞了一個通用模板

開發 后端
最近工作中,我通過層層優化重復代碼,最后抽出個通用模板.因此跟大家分享一下優化以及思考的過程。我會先造一個相似的例子,然后一步步帶大家如何優化哈,看完一定會有幫助的。

后端思維

大家好,我是田螺。

后端思維系列好久沒更新啦~今天,終于來了。本文是田螺哥后端思維專欄的第7篇哈。

最近工作中,我通過層層優化重復代碼,最后抽出個通用模板.因此跟大家分享一下優化以及思考的過程。我會先造一個相似的例子,然后一步步帶大家如何優化哈,看完一定會有幫助的。

  • 優化前的例子
  • 第一步優化:抽取公用方法
  • 第二步優化:反射對比字段
  • 第三步優化:泛型+ lambda函數式
  • 第四步優化:繼承多態
  • 第五步優化:模板方法成型
  • 大功告成:  策略模式+工廠模式+模板方法模式

1. 優化前的例子

在這里,我先給大家模擬一個業務場景哈,并給出些簡化版的代碼

假設你有個對賬需求:你要把文件服務器中,兩個A、B不同端,上送的余額明細和轉賬明細,下載下來,對比每個字段是否一致.

明細和余額的對比類似,代碼整體流程:

  • 讀取A、B端文件到內存的兩個list
  • 兩個list通過某個唯一key轉化為map
  • 兩個map字段逐個對比

我們先看明細對比哈,可以寫出類似醬紫的代碼:

//對比明細
private void checkDetail(String detailPathOfA,String detailPathOfB )throws IOException{

   //讀取A端的文件
   List<DetailDTO> resultListOfA = new ArrayList<>();
   try (BufferedReader reader1 = new BufferedReader(new FileReader(detailPathOfA))) {
            String line;
            while ((line = reader1.readLine()) != null) { 
                resultListOfA.add(DetailDTO.convert(line));
            }
        }
    
   //讀取B端的文件
   List<DetailDTO> resultListOfB = new ArrayList<>();
   try (BufferedReader reader1 = new BufferedReader(new FileReader(detailPathOfB))) {
            String line;
            while ((line = reader1.readLine()) != null) { 
                resultListOfB.add(DetailDTO.convert(line));
            }
        }
    
    //A列表轉化為Map
    Map<String,DetailDTO> resultMapOfA = new HashMap<>();
    for(DetailDTO detail:resultListOfA){
        resultMapOfA.put(detail.getBizSeq(),detail);
    }
    
     //B列表轉化為Map
    Map<String,DetailDTO> resultMapOfB = new HashMap<>()
    for(DetailDTO detail:resultListOfB){
        resultMapOfB.put(detail.getBizSeq(),detail);
    }
    
    //明細逐個對比
    for (Map.Entry<String, DetailDTO> temp : resultMapOfA.entrySet()) {
        if (resultMapOfB.containsKey(temp.getKey())) {
            DetailDTO detailOfA = temp.getValue();
            DetailDTO detailOfB = resultMapOfB.get(temp.getKey());

            if (!detailOfA.getAmt().equals(detailOfB.getAmt())) {
                  log.warn("amt is different,key:{}", temp.getKey());
            }
            if (!detailOfA.getDate().equals(detailOfB.getDate())) {
                log.warn("date is different,key:{}", temp.getKey());
            }

            if (!detailOfA.getStatus().equals(detailOfB.getStatus())) {
                log.warn("status is different,key:{}", temp.getKey());
            }
            ......
        }
  }
}

2. 抽取公用方法去重

大家仔細看以上明細對比的例子,發現了重復代碼:

圖片圖片

我們可以抽取一個公用方法去優化它,比如抽取個讀取文件的公用方法 readFile:

//對比明細
private void checkDetail(String detailPathOfA,String detailPathOfB )throws IOException{

   //讀取A端的文件
    List<DetailDTO> resultListOfA = readFile(detailPathOfA);
   //讀取B端的文件
   List<DetailDTO> resultListOfB = readFile(detailPathOfB);
   ......
}

//抽取公用方法
 private List<DetailDTO> readFile(String detailPath) throws IOException {
        List<DetailDTO> resultList = new ArrayList<>();
        try (BufferedReader reader1 = new BufferedReader(new FileReader(detailPath))) {
            String line;
            while ((line = reader1.readLine()) != null) {
                resultList.add(DetailDTO.convert(line));
            }
        }
        return resultList;
    }

同理,這塊代碼也是重復了:

圖片圖片

我們也可以抽個公用方法:convertListToMap

//對比明細
private void checkDetail(String detailPathOfA,String detailPathOfB ){

   //讀取A端的文件
    List<DetailDTO> resultListOfA = readFile(detailPathOfA);
   //讀取B端的文件
   List<DetailDTO> resultListOfB = readFile(detailPathOfB);
   
   //A列表轉化為Map
   Map<String,DetailDTO> resultMapOfA = convertListToMap(resultListOfA);
   //B列表轉化為Map
   Map<String,DetailDTO> resultMapOfB = convertListToMap(resultListOfB);
   ......
}

//抽取公用方法
private Map<String,DetailDTO> convertListToMap(List<DetailDTO> list){
    Map<String,DetailDTO> map = new HashMap<>()
    for(DetailDTO detail:list){
        map.add(detail.getBizSeq(),detail);
    }
    return map;
}

通過抽取公用方法后,已經優雅很多啦~

3. 反射對比字段

我們再來看下字段對比的邏輯,如下:

圖片圖片

以上代碼會取兩個對象的每個字段對比,如果明細對象的屬性字段特別多的話,這塊代碼也會顯得重復冗余。我們可以通過反射去對比兩個對象的屬性,如下:

public static List<String> compareObjects(Object obj1, Object obj2) {
        List<String> list = new ArrayList<>();

        Class<?> clazz = obj1.getClass();
        Field[] fields = clazz.getDeclaredFields();

        for (Field field : fields) {
            String fieldName = field.getName();
            field.setAccessible(true);
            try {
                Object value1 = field.get(obj1);
                Object value2 = field.get(obj2);

                if ((value1 == null && value2 != null) || (value1 != null && !value1.equals(value2))) {
                    list.add(fieldName);
                }
            } catch (IllegalAccessException e) {
                e.printStackTrace();
            }
        }
        return list;
        }

有了這個反射對比方法,原來的代碼就可以優化成這樣啦,是不是優雅了很多:

//對比明細
private void checkDetail(String detailPathOfA,String detailPathOfB ){

   //讀取A端的文件
    List<DetailDTO> resultListOfA = readFile(detailPathOfA);
   //讀取B端的文件
   List<DetailDTO> resultListOfB = readFile(detailPathOfB);
   
   //A列表轉化為Map
   Map<String,DetailDTO> resultMapOfA = convertListToMap(resultListOfA);
   //B列表轉化為Map
   Map<String,DetailDTO> resultMapOfB = convertListToMap(resultListOfB);
   
    //明細逐個對比
    for (Map.Entry<String, DetailDTO> temp : resultMapOfA) {
          if(resultMapOfB.containsKey(temp.getKey()){
             DetailDTO detailOfA = temp.getValue();
             DetailDTO detailOfB = resultMapOfB.get(temp.getKey());
             
             List<String> resultList=compareObjects(detailOfA,detailOfB);
             for(String temp:resultList){
                log.warn("{} is different,key:{}",temp,detailOfA.getKey()); 
             }
            ......
          }
      }
}

4.Lambda函數式+泛型

實現完明細文件的對比,我們還需要余額文件的對比:

同樣的,也是先讀取文件,如下:

//對比明細
private void checkBalance(String balancePathOfA,String balancePathOfB ){

   //讀取A端的文件
   List<BalanceDTO> resultListOfA = new ArrayList<>();
   try (BufferedReader reader1 = new BufferedReader(new FileReader(balancePathOfA))) {
            String line;
            while ((line = reader1.readLine()) != null) { 
                resultListOfA.add(BalanceDTO.convert(line));
            }
        }
        
   List<DetailDTO> resultListOfB = new ArrayList<>();
   try (BufferedReader reader1 = new BufferedReader(new FileReader(detailPathOfB))) {
            String line;
            while ((line = reader1.readLine()) != null) { 
                resultListOfB.add(DetailDTO.convert(line));
            }
        }
    ......
    }

大家可以發現,讀取余額文件和剛剛的讀取明細文件很像,有一部分代碼是重復的,但是不能直接一下子抽個共同函數出來:

圖片圖片

對了,convert方法是醬紫的哈:

public static BalanceDTO convert(String line){
        BalanceDTO dto = new BalanceDTO();
        String[] dataLine = line.split(",",-1);
        dto.setBalance(dataLine[1]);
        dto.setType(dataLine[2]);
        ......
        return dto;
    }

大家可以發現,就是一個返回類型,以及這個對應類型的一個靜態convert方法不一致而已,如果是類型不一樣,我們可以使用泛型替代,如果是一個小的靜態方法不一致,我們則可以使用lambda函數式接口提取,因此可以抽這個這么一個公用方法吧:

public <T> List<T> readDataFromFile(String filePath, Function<String, T> converter) throws IOException {
    List<T> result = new ArrayList<>();
    try (BufferedReader reader = new BufferedReader(new FileReader(filePath))) {
        String line;
        while ((line = reader.readLine()) != null) { 
            result.add(converter.apply(line));
        }
    }
    return result;
}

//余額讀取調用
List<BalanceDTO> resultListOfA = readDataFromFile(balancePathOfA, BalanceDTO::convert);
//明細讀取調用
List<DetailDTO> resultList = readDataFromFile(detailPath, DetailDTO::convert);

平時我們用泛型+ Lambda表達式結合,去抽取公用方法,代碼就顯得高端大氣很多,對吧!

5. 繼承多態.

在余額對比文件中,讀取完文件到內存后,我們需要把通過某個唯一key關聯起來,即把List轉為Map,如下:

//對比明細
private void checkBalance(String balancePathOfA,String balancePathOfB ){

  //讀取A端的文件
  List<BalanceDTO> resultListOfA = readDataFromFile(balancePathOfA, BalanceDTO::convert);
   //讀取B端的文件
  List<BalanceDTO> resultListOfB = readDataFromFile(balancePathOfB, BalanceDTO::convert);
   
  //A列表list轉化為Map
  Map<String,BalanceDTO> resultMapOfA = new HashMap<>()
  for(BalanceDTO balance:resultListOfA){
    resultMapOfA.add(balance.getType()+balance.getAccountNo(),balance);
  }

一般來說,把兩個list轉化為Map,抽一個公用方法是不是就好了?比如說醬紫:

private Map<String,BalanceDTO> convertListToMap(List<BalanceDTO> list){
    Map<String,BalanceDTO> map = new HashMap<>()
    for(BalanceDTO balance:list){
        resultMapOfA.add(balance.getType()+balance.getAccountNo(),balance);
    }
    return map;
}

其實也行,但是其實可以更抽象一點。因為余額和明細對比都有list轉map的需求,而且也是有共性的,只不過是轉化map的key和value的類型不一致而已。

圖片圖片

我們仔細思考一下,value類型是不同類型(分別是BalanceDTO和DetailDTO),而key則是對應對象的一個或者某幾個屬性連接起來的。對于不同類型,我們可以考慮泛型。對于余額和明細對象不同的key的話,我們則可以考慮繼承和多態,讓它們實現同一個接口就好啦。

我們可以使用繼承和多態,定義一個抽象類BaseKeyDTO,里面有個getKey的抽象方法,然后BalanceDTO 和DetailDTO都繼承它,實現各自getKey的方法,如下:

public abstract class BaseDTO {
    abstract String getKey();
} 

public class BalanceDTO extends BaseDTO {
    @Override
    String getKey() {
        return type + accountNo;
    }
}

public class DetailDTO extends BaseDTO {
    @Override
    String getKey() {
        return bizSeq;
  }

最后,我們應用繼承多態+擴展泛型(<T extends BaseDTO>),就可以把余額和明細對比的convertListToMap方法抽成一個啦:

private static <T extends BaseDTO> Map<String, T> convertListToMap(List<T> list) {
        Map<String, T> map = new HashMap<>();
        for (T item : list) {
            map.put(item.getKey(), item);
        }
        return map;
    }

最后明細和余額對比,可以優化成這樣,其實看起來已經比較優雅啦:

//對比明細
    private void checkDetail(String detailPathOfA, String detailPathOfB) throws IOException {

        //讀取A端明細的文件
        List<DetailDTO> resultListOfA = readDataFromFile(detailPathOfA, DetailDTO::convert);
        //讀取B端明細的文件
        List<DetailDTO> resultListOfB = readDataFromFile(detailPathOfB, DetailDTO::convert);

        //A列表轉化為Map
        Map<String, DetailDTO> resultMapOfA = convertListToMap(resultListOfA);
        //B列表轉化為Map
        Map<String, DetailDTO> resultMapOfB = convertListToMap(resultListOfB);

        //明細逐個對比
        compareDifferent(resultMapOfA,resultMapOfB);
    }
    

   //對比余額
    private void checkBalance(String balancePathOfA,String detailPathOfB) throws IOException {

        //讀取A端余額的文件
        List<BalanceDTO> resultListOfA = readDataFromFile(balancePathOfA,BalanceDTO::convert);
        //讀取B端余額的文件
        List<BalanceDTO> resultListOfB = readDataFromFile(detailPathOfB,BalanceDTO::convert);

        //A余額列表轉化為Map
        Map<String,BalanceDTO> resultMapOfA = convertListToMap(resultListOfA);
        //B余額列表轉化為Map
        Map<String,BalanceDTO> resultMapOfB = convertListToMap(resultListOfB);

        //余額逐個對比
        compareDifferent(resultMapOfA,resultMapOfB);
    }
    
    //對比也用泛型,抽一個公用的方法哈
    private void compareDifferent(Map<String, T> mapA, Map<String, T> mapB) {
        for (Map.Entry<String, T> temp : mapA.entrySet()) {
            if (mapB.containsKey(temp.getKey())) {
                T dtoA = temp.getValue();
                T dtoB = mapB.get(temp.getKey());

                List<String> resultList = compareObjects(dtoA, dtoB);
                for (String tempStr : resultList) {
                    log.warn("{} is different,key:{}", tempStr, dtoA.getKey());
                }
            }
        }
    }
}

6. 模板方法

大家回頭細看,可以發現不管是明細還是余額對比,兩個方法很像,都是一個骨架流程來的:

  • 讀取A、B端文件到內存的兩個list
  • 兩個list通過某個唯一key轉化為map
  • 兩個map字段逐個對比

圖片圖片

大家先回想一下模板方法模式:

定義了一個算法的骨架,將一些步驟延遲到子類中實現。這有助于避免在不同類中重復編寫相似的代碼。

頓時是不是就覺得這塊代碼還有優化空間。

6.1 定義對比模板的骨架

我們可以嘗試這兩塊代碼再合并,用模板方法優化它。我們先定義一個模板,然后模板內定義它們骨架的流程,如下:

//聲明對比抽象模板
public abstract class AbstractCheckTemplate<T extends BaseDTO> {

    public void checkTemplate(String filePathA, String filePathB) throws IOException {

        //從文件讀取為List
        readDataFromFile(filePathA, filePathB);
        //list轉化為Map
        covertListToMap(resultListOfA, resultListOfB);
        //比較
        compareDifferent(mapA, mapB);
    }

6.2 模板的方法逐步細化

因為readDataFromFile需要輸出兩個list,所以我們可以定義返回類型為Pair,代碼如下:

private Pair<List<T>, List<T>> readDataFromFile(String filePathA, String filePathB, Function<String, T> converter) throws IOException {
        //讀取A端余額的文件
        List<T> resultListOfA = readDataFromFile(filePathA, converter);
        //讀取B端余額的文件
        List<T> resultListOfB = readDataFromFile(filePathB, converter);
        return new Pair<>(resultListOfA, resultListOfB);
    }

又因為這個函數式的轉化,是不同子類才能定下來的,我們就可以聲明個抽象方法convertLineToDTD,讓子類去實現。因此模板就變成這樣啦:

public abstract class AbstractCheckTemplate<T extends BaseDTO> {

    public void checkTemplate(String filePathA, String filePathB) throws IOException {

        //從文件讀取為List
        Pair<List<T>, List<T>> resultListPair = readDataFromFile(filePathA, filePathB, this::convertLineToDTD);
        List<T> resultListOfA = resultListPair.getKey();
        List<T> resultListOfB = resultListPair.getValue();

        //list轉化為Map
        covertListToMap(resultListOfA, resultListOfB);
        //比較
        compareDifferent(mapA, mapB);
    }
    
    //延遲到子類實現轉換為不同的DTO
    protected abstract T convertLineToDTD(String line);

同理,還有兩個list轉化為兩個map再對比,我們可以聲明為這樣:

private Pair<Map<String, T>, Map<String, T>> covertListToMap(List<T> listA, List<T> listB) {
        return new Pair<>(convertListToMap(listA), convertListToMap(listB));
    }

因此最終模板就是這樣啦:

@Slf4j
public abstract class AbstractCheckTemplate<T extends BaseDTO> {

    public void checkTemplate(String filePathA, String filePathB) throws IOException {

        //從文件讀取為List
        Pair<List<T>, List<T>> resultListPair = readDataFromFile(filePathA, filePathB, this::convertLineToDTD);
        List<T> resultListOfA = resultListPair.getKey();
        List<T> resultListOfB = resultListPair.getValue();

        //list轉化為Map
        Pair<Map<String, T>, Map<String, T>> resultMapPair = covertListToMap(resultListOfA, resultListOfB);
        Map<String, T> mapA = resultMapPair.getKey();
        Map<String, T> mapB = resultMapPair.getValue();

        //比較
        compareDifferent(mapA, mapB);
    }
    
    protected abstract T convertLineToDTD(String line);
    ......此處省略公用的私有方法
}

6.3 不同對比子類

如果你是余額對比,那你聲明一個CheckBalanceStrategyServiceImpl去繼承抽象模板。

/**
 * 余額對比策略
 * 公眾號: 撿田螺的小男孩
 */
@Service
public class CheckBalanceStrategyServiceImpl extends AbstractCheckTemplate<BalanceDTO> {

    @Override
    protected BalanceDTO convertLineToDTD(String line) {
        return BalanceDTO.convert(line);
    }
}

如果你是明細對比,那你聲明一個CheckDetailStrategyServiceImpl去繼承抽象模板。

/**
 * 明細對比策略
 * 關注公眾號: 撿田螺的小男孩
 */
@Service
public class CheckDetailStrategyServiceImpl extends AbstractCheckTemplate<DetailDTO> {
    @Override
    protected DetailDTO convertLineToDTD(String line) {
        return DetailDTO.convert(line);
    }
}

這兩個不同的子類,就像不同的策略,我們應該都能嗅到策略模式的味道啦~

7. 工廠模式+ 模板方法 + 策略模式全家桶

有了明細對比、余額對比的模板,為了更方便調用,我們還可以定義一個校驗策略接口,然后交給spring工廠類,這樣更方便調用。其實日常開發中,這三種設計模式一般一起出現,非常實用:

我們先聲明一個校驗ICheckStrategy接口:

/**
 * 關注公眾號: 撿田螺的小男孩
 */
public interface ICheckStrategy {

    /**
     * 對比校驗邏輯
     * @param filePathA
     * @param filePathB
     * @throws IOException
     */
    void check(String filePathA, String filePathB) throws IOException;

    /**
     * 校驗的類型,明細/余額
     * @return
     */
    CheckEnum getCheckEnum();
}

然后,模板AbstractCheckTemplate實現ICheckStrategy接口。

public abstract class AbstractCheckTemplate<T extends BaseDTO> implements ICheckStrategy {

接著,不同對比策略類CheckDetailStrategyServiceImpl 和CheckDetailStrategyServiceImpl映射對應的對比校驗類型:

/**
 * 明細對比策略
 * 關注公眾號: 撿田螺的小男孩
 */
@Service
public class CheckDetailStrategyServiceImpl extends AbstractCheckTemplate<DetailDTO> {

     @Override
    protected DetailDTO convertLineToDTD(String line) {
        return DetailDTO.convert(line);
    }
  
    @Override
    public void check(String filePathA, String filePathB) throws IOException {
        checkTemplate(filePathA, filePathB);
    }

    //對比校驗類型為:明細
    @Override
    public CheckEnum getCheckEnum() {
        return CheckEnum.DETAIL_CHECK;
    }
}

/**
 * 余額對比策略
 * 關注公眾號: 撿田螺的小男孩
 */
@Service
public class CheckBalanceStrategyServiceImpl extends AbstractCheckTemplate<BalanceDTO> {

    @Override
    public void check(String filePathA, String filePathB) throws IOException {
        checkTemplate(filePathA, filePathB);
    }
     //對比校驗類型為:余額
    @Override
    public CheckEnum getCheckEnum() {
        return CheckEnum.BALANCE_CHECK;
    }

    @Override
    protected BalanceDTO convertLineToDTD(String line) {
        return BalanceDTO.convert(line);
    }
}

最后一步,我們借助spring的生命周期,使用ApplicationContextAware接口,把對用的策略,初始化到map里面。然后對外提供checkCompare方法即可。讓調用者決定用哪一種對比,其實這算工廠模式思想,大家可以自己思考一下~

@Component
public class CheckCompareFactory implements ApplicationContextAware {

    private final Map<CheckEnum, ICheckStrategy> checkStrategyMap = new ConcurrentHashMap<>();
    
    //把不同策略放到map
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        Map<String, ICheckStrategy> tmepMap = applicationContext.getBeansOfType(ICheckStrategy.class);
        tmepMap.values().forEach(strategyService -> checkStrategyMap.put(strategyService.getCheckEnum(), strategyService));
    }

    /**
     * 直接調用這個方法即可
     */
    public void checkCompare(CheckEnum checkEnum, String filePathA, String filePathB) throws IOException {
        ICheckStrategy checkStrategy = checkStrategyMap.get(checkEnum);
        checkStrategy.check(filePathA, filePathB);
    }
}

最后

我是撿田螺的小男孩。本文介紹了:如何將一些通用的、用于優化重復冗余代碼的技巧應用到開發中。最終,我通過這些技巧將代碼優化成一個通用模板。

責任編輯:武曉燕 來源: 撿田螺的小男孩
相關推薦

2024-04-07 00:00:01

TypeScript語言REST

2025-11-04 07:20:00

Vue前端開發

2024-01-22 09:28:27

2022-06-20 08:15:11

后端觀察者模板

2025-11-13 10:07:08

2022-05-18 08:51:44

調用模板后端并行

2022-03-23 18:00:34

循環CPU線程

2022-09-05 19:00:53

低代碼平臺React

2019-08-22 17:19:19

javascript去重數組

2013-12-25 10:24:15

創業應用開發

2021-10-14 10:05:17

開源項目代碼

2021-08-04 17:40:42

代碼Java消息冪等

2024-04-10 07:56:38

前端數組uniq

2022-03-07 05:53:41

線程CPU代碼

2022-07-04 07:37:51

模板模式重構

2009-06-22 13:50:00

java連接mysql

2022-02-18 09:20:43

消息中間件分布式MQ 冪等

2022-05-20 08:09:18

設計模式后端代碼

2019-05-23 08:55:41

代碼開發工具

2018-04-25 08:45:46

大數據
點贊
收藏

51CTO技術棧公眾號

免费观看黄色一级视频| 乳色吐息在线观看| 91av资源在线| 久久国产精品第一页| 欧美裸体男粗大视频在线观看| 无码人妻一区二区三区免费n鬼沢 久久久无码人妻精品无码 | 在线视频精品| 色婷婷综合成人| 波多野结衣一二三区| 黄色成人小视频| 亚洲成人午夜影院| 一区二区三区四区五区视频 | 亚洲天堂成人在线| 免费不卡av网站| 在线观看精品| 午夜日韩在线观看| 亚洲最新免费视频| 深夜福利在线视频| 国产一区激情在线| 国产精品久久久久久久久久小说| 久久久夜色精品| 日韩伦理一区| 日韩av在线免费播放| 久久精品一卡二卡| 成人黄色免费网站| 欧美香蕉大胸在线视频观看| av动漫在线播放| 免费av网站在线看| 国产欧美日韩卡一| 久精品国产欧美| 亚洲精华国产精华精华液网站| 青青草国产成人av片免费| 性欧美办公室18xxxxhd| 中文字幕影音先锋| 国产精品国产一区| 伊人亚洲福利一区二区三区| 大地资源二中文在线影视观看| 亚洲精品v亚洲精品v日韩精品| 欧美人狂配大交3d怪物一区 | 国产九九视频一区二区三区| 国产精品一二三在线| 精产国品一区二区| 美女爽到呻吟久久久久| 97欧美精品一区二区三区| 青娱乐国产在线| 综合久久综合| 欧美成人h版在线观看| 日韩av网站在线播放| 成人在线免费观看网站| 亚洲性猛交xxxxwww| 97人妻精品一区二区三区免| 成人激情自拍| 亚洲第一精品夜夜躁人人躁| 精品少妇人妻av一区二区三区| 日韩精品成人在线观看| 欧美一级艳片视频免费观看| 佐山爱在线视频| 精品入口麻豆88视频| 欧美一区二区三区在线看 | 91在线国产观看| 精品国产免费人成电影在线观... 精品国产免费久久久久久尖叫 | 国产精品乱码一区二区三区| 国精产品乱码一区一区三区四区| 成人免费看视频| 国产伦精品一区二区三毛| 色婷婷av一区二区三区之红樱桃| av在线不卡电影| 麻豆亚洲一区| 国产黄色片在线播放| 国产欧美日韩视频一区二区 | 精品动漫3d一区二区三区免费版| 欧美精品久久久久a| 欧美福利视频一区二区| 麻豆亚洲精品| 国产日韩av在线| 性一交一乱一色一视频麻豆| 99久久免费国产| 任我爽在线视频精品一| 日本在线观看www| 亚洲人成在线观看一区二区| 日韩av新片网| 日韩制服一区| 欧美一区二区三区爱爱| 2一3sex性hd| av影片在线一区| 欧美成人免费在线视频| 99久在线精品99re8热| 首页国产欧美日韩丝袜| 91视频国产高清| 亚洲 欧美 激情 小说 另类| 国产精品天天摸av网| 伊人再见免费在线观看高清版| 美女网站在线看| 欧美老人xxxx18| 亚洲av无码一区二区三区观看| 成人区精品一区二区婷婷| 久久综合免费视频影院| 国产精品黄色网| 久久成人18免费观看| 国产福利久久精品| aⅴ在线视频男人的天堂 | 在线视频精品免费| 国产成人自拍在线| 天天爽天天狠久久久| 成人爽a毛片免费啪啪动漫| 色8久久精品久久久久久蜜| 久久久久久国产精品日本| 蜜臀91精品国产高清在线观看| 久久夜色精品国产亚洲aⅴ| 亚洲va在线观看| 成人一二三区视频| 在线视频不卡一区二区| 自拍网站在线观看| 欧美www视频| 欧美特黄一级片| 首页综合国产亚洲丝袜| 精品视频一区在线| 任你弄在线视频免费观看| 欧美日韩中文精品| 久久久久亚洲av无码专区桃色| 国产精品观看| 91精品国产综合久久香蕉的用户体验| 欧美孕妇孕交| 无码av中文一区二区三区桃花岛| 天天干天天曰天天操| 欧美日韩久久精品| 全亚洲最色的网站在线观看| 欧美一级淫片aaaaaa| 亚洲欧美日韩中文播放| 天天干天天玩天天操| 欧美美女在线| 欧美一区第一页| 欧洲精品久久一区二区| 一二三区精品视频| 少妇愉情理伦片bd| 久久久国产精品| 成人av在线亚洲| 日韩大片在线永久免费观看网站| 在线视频综合导航| 女人又爽又黄免费女仆| 亚洲专区免费| 久久精品一二三区| 天堂av中文在线观看| 亚洲成人黄色网| 日本中文字幕免费观看| 成人午夜视频网站| 精品少妇在线视频| 韩国精品福利一区二区三区| 韩国精品久久久999| 老牛影视av牛牛影视av| 亚洲午夜影视影院在线观看| 少妇高潮一69aⅹ| 黄色日韩在线| 国精产品一区二区| 在线天堂中文资源最新版| 日韩电影免费在线观看中文字幕| wwwxxx亚洲| 久久久激情视频| 国产免费又粗又猛又爽| 婷婷综合亚洲| 91嫩草在线| freexxx性亚洲精品| 亚洲精品国精品久久99热一| 黄色在线免费观看| 中文字幕精品一区二区三区精品| 一本色道久久亚洲综合精品蜜桃 | 日韩在线国产精品| 国产男女无套免费网站| 亚洲一区影音先锋| 国产白嫩美女无套久久| 日产欧产美韩系列久久99| 亚洲综合欧美日韩| 日韩精品一区国产| 91精品国产乱码久久久久久蜜臀| 蜜桃视频在线观看网站| 欧美日韩一区二区三区视频| 强乱中文字幕av一区乱码| 99久久久精品| 国产精品人人爽人人爽| 亚洲成人av| 国产一区二区自拍| 国产精品亚洲成在人线| 欧美美最猛性xxxxxx| 婷婷综合激情网| 欧美日韩高清影院| 香蕉视频一区二区| 国产精品网友自拍| 人妖粗暴刺激videos呻吟| 日日夜夜精品免费视频| 国产午夜精品视频一区二区三区| 日本久久成人网| 92国产精品视频| 91精品论坛| 九九精品在线视频| 国产一区二区三区不卡在线| 日韩欧美激情一区| 天天干天天操天天操| 一区二区三区精品在线观看| 乐播av一区二区三区| 丰满少妇久久久久久久| 国产高潮免费视频| 99国产精品| 国产日韩第一页| 久久综合亚洲| 国产综合精品一区二区三区| 国产美女久久| 欧美中文字幕在线观看| 亚洲资源一区| 色噜噜狠狠狠综合曰曰曰| 性xxxx搡xxxxx搡欧美| 日韩一区二区三区视频在线观看| 欧美brazzers| 欧美视频不卡中文| 国产在线拍揄自揄拍| 中文字幕一区二区三区乱码在线| 亚洲午夜福利在线观看| 成人午夜免费电影| 天天做天天干天天操| 久久最新视频| 欧美国产激情视频| 亚洲欧洲日本mm| 国产91视频一区| 亚洲精品2区| 亚洲一区在线免费| 日本一区二区免费高清| 日韩国产在线一区| 亚欧洲精品视频在线观看| 国产精品一区二区三区观看| 精品国产麻豆| 91日本视频在线| 国外成人福利视频| 国产精品欧美激情在线播放| 亚洲a∨精品一区二区三区导航| 69久久夜色精品国产69| segui88久久综合| 久久久久久久久久久久av| 中文字幕在线观看播放| 免费97视频在线精品国自产拍| 中文字幕在线视频区| 在线一区二区日韩| 你懂的好爽在线观看| 亚洲乱亚洲乱妇无码| 三级理论午夜在线观看| 日韩av一区在线观看| 日韩大胆人体| 亚洲欧洲在线看| 国产免费av在线| 中文字幕日韩av综合精品| 北条麻妃在线| 色视频www在线播放国产成人| 婷婷成人激情| 乱亲女秽乱长久久久| 在线中文字幕-区二区三区四区| 欧美成人三级视频网站| gratisvideos另类灌满| 欧美国产日韩一区二区三区| rebdb初裸写真在线观看| 97成人在线视频| 电影网一区二区| 国产精品久久久久久五月尺| 亚洲黑人在线| 亚洲伊人一本大道中文字幕| jizz久久精品永久免费| 精品久久久久久综合日本| 亚洲宅男一区| 在线视频精品一区| 国产精品va| 日本在线观看a| 免费高清成人在线| avtt中文字幕| 久久久久99精品一区| 美女av免费看| 亚洲一区二区三区在线看| 91看片在线播放| 欧美视频一区二区三区四区 | 国产不卡精品| 日韩一区二区三区视频| 国产xxxx在线观看| 精品久久一二三区| 五月天婷婷视频| 成人欧美精品一区二区| 久久亚洲欧洲| 粉色视频免费看| 国产精品乡下勾搭老头1| 国产高清成人久久| 国产午夜精品一区二区三区嫩草| 国产又粗又猛又爽又黄的视频小说| 亚洲精品欧美专区| 久久99精品波多结衣一区| 欧美日韩在线播放一区| 精品国产av 无码一区二区三区| 亚洲成人精品视频| av天在线观看| 久久久久久尹人网香蕉| 国产成人免费9x9x人网站视频| 2014国产精品| 精品国产一区二区三区四区| 国产爆乳无码一区二区麻豆 | 日本一区二区三区四区| 一级黄色录像视频| 91精品福利在线| 亚洲精品国产suv一区| 一区二区三区四区精品| 2021中文字幕在线| 成人黄色片网站| 视频一区欧美| 热99这里只有精品| 国产在线精品一区二区| 中文字幕av网址| 亚洲国产精品精华液网站| 一区两区小视频| 亚洲老头老太hd| 国产精品一品| 亚洲japanese制服美女| 狠狠做六月爱婷婷综合aⅴ| av网站手机在线观看| 精品一区中文字幕| 乐播av一区二区三区| 精品福利樱桃av导航| 亚洲成人77777| 久久久精品久久| 国产精品无码久久久久| 欧美一区二区在线视频观看| 9国产精品视频| 亚洲一区和二区| 亚洲人成精品久久久久久| 亚洲男人天堂网址| 亚洲精品一区二区在线| 电影在线观看一区| av成人免费观看| 亚洲第一天堂| 亚洲天堂av一区二区三区| 中文字幕不卡在线| 亚洲精品国产欧美在线观看| 亚洲人成电影网站色xx| 中文字幕21页在线看| 精品一卡二卡三卡四卡日本乱码 | 亚洲qvod图片区电影| 97精品一区二区| 亚洲人视频在线| 国产精品传媒入口麻豆| 中文字字幕在线观看| 曰本色欧美视频在线| 成人午夜sm精品久久久久久久| 欧美不卡在线一区二区三区| 亚洲一区不卡| av女人的天堂| 欧美亚洲高清一区二区三区不卡| 大乳在线免费观看| 国产精品吊钟奶在线| 欧美另类69xxxxx| 久久久久久久久久久久91| 国产精品毛片a∨一区二区三区| 日本一区二区三区久久| 日韩网站在线观看| 午夜精品久久久久久毛片| 特级黄色录像片| 国产成人亚洲综合a∨猫咪| 久久久国产精华液| 日韩精品视频在线播放| 欧美日韩美女| 一区二区冒白浆视频| 国产精一区二区三区| 日本熟妇成熟毛茸茸| 日韩精品亚洲精品| 国产麻豆久久| 看一级黄色录像| 成人av网在线| 中文字幕一区在线播放| 色天天综合狠狠色| 91成人入口| 99热成人精品热久久66| 国产精品色眯眯| 亚洲免费不卡视频| 亲子乱一区二区三区电影| 久久精品国产大片免费观看| 99视频在线观看视频| 激情懂色av一区av二区av| а天堂8中文最新版在线官网| 亚洲va久久久噜噜噜久久天堂| 亚洲免费播放| 国产精品1区2区3区4区| 精品美女一区二区三区| 97久久香蕉国产线看观看| 国产精品无码乱伦| av激情综合网| 91在线观看喷潮| 7777精品视频| 66视频精品| 中文字幕av观看| 91精品蜜臀在线一区尤物| 国产在线美女| 在线观看国产一区| 99久久综合精品| 国产又粗又猛又爽| 4p变态网欧美系列| 女生裸体视频一区二区三区| 亚洲av无码一区二区二三区| 日韩一级完整毛片| 午夜无码国产理论在线|