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

通過Python腳本支持OC代碼重構實踐:模塊調用關系分析

開發 前端
在軟件開發中,經常會遇到一些代碼問題,例如邏輯結構復雜、依賴關系混亂、代碼冗余、不易讀懂的命名等。這些問題可能導致代碼的可維護性下降,增加維護成本,同時也會影響到開發效率。

在軟件開發中,經常會遇到一些代碼問題,例如邏輯結構復雜、依賴關系混亂、代碼冗余、不易讀懂的命名等。這些問題可能導致代碼的可維護性下降,增加維護成本,同時也會影響到開發效率。

這時通常通過重構的方式,在不改變軟件的功能和行為的前提下,對軟件的代碼進行重新組織和優化。達到增強代碼的可讀性,降低維護成本,提升研發效率和質量的目的。通過合理的重構,可以大大提高軟件的可維護性和可擴展性,從而延長其生命。

本系列的內容介紹了百度App搜索側業務如何使用Python腳本實現自動化工具,以支持百度App配置數據項調用方式升級為數據通路的重構過程。通過Python腳本,我們實現一些自動化的工具,包括配置數據項調用關系分析、配置數據項接入數據通路的實現、數據項使用方接入數據通路的適配等,以期提高工作效率、減少出錯率。

01、代碼重構時的關鍵步驟及挑戰

在代碼重構過程中,需要考慮重構的效率和重構后的代碼質量。與其相關的關鍵的步驟如下,這些步驟先后依賴,相互影響:

熟悉業務及技術現狀:在開始重構之前,研發首先要理解業務邏輯和流程,熟悉業務能力及技術實現時存在的問題,確定重構的范圍。

確定重構方案:基于對業務邏輯和現有代碼問題的理解,確定重構方案,重點關注有兩點,有問題的代碼如何重構和依賴于該代碼的調用如何適配。

分階段實施:根據重構方案,分階段的修改代碼,并測試代碼的功能是否正常。在修改過程中,應該盡量避免影響到不相關模塊,這樣可以更好地控制風險。

效果評估及監控:重構方案開發完成,線下對實現的效果進行評估,線上對實現的效果進行監控,及時發現異常止損和重構的效果。

在重構的工作中,大部分的工作是人工的方式完成,是一個耗時且容易出錯的過程。對于研發人員來講,在不改變軟件的功能和行為的前提下,保證質量和效率完成對已有功能的重構,是一個極大的挑戰。

02、百度App(iOS)搜索側的配置數據項重構

為了更好的提升系統穩定性和降低配置數據項變更時對上層依賴方組件的影響, 我們決定對百度App(iOS)搜索側的配置數據項進行重構。重構過程的關鍵節點中有超過80%的工作是由自動化工具完成,支持重構工作上線后零bug,和全部的配置數據項接口內斂,提升了系統的安全性和穩定性。

2.1 重構背景

百度App(iOS)-搜索側的配置數據項,大部分集中在一個類(XXXSetting)中管理。該類(XXXSetting)以獨立組件的方式發布,被超過30個其它組件依賴。

如圖-1 所示數據項使用模塊直接調用數據項提供模塊(XXXSetting),是直接依賴的關系,數據項的增刪相當于接口的變更,對上層的依賴方會產生影響,當接口存在不兼容變更時,連帶上層的依賴方組件也需要二次的發布。且該組件中的數據項主要為實驗類開關,變動較為頻繁,影響面也被放大,故需使用比較穩定的方式實現不同模塊之間的數據項共享。

圖-1圖-1

2.2 技術方案

在技術實現的層面,主要分為兩步

1、第一步為實現多模塊之間數據通訊的模塊,在本系列的內容中以數據通路代指該模塊。

2、第二步為基于數據通路提供的能力,XXXSetting組件為作數據提供方接入數據通路,原使用XXXSetting組件的使用方接入數據通路,這樣就完成了XXXSetting組件中的數據項遷移。

數據通路的實現,目標實現以Key-Value的方式讀取及更新配置項,需要從無到有的構建,在本系列的其它章節中內容會有介紹。但XXXSetting組件對應的重構工作,是基于已有的線上能力的改造,Setting中的數據項超過百個,外部的調用點也是以百為計算單位,涉及的組件有30+。影響面如何評估,如何保證重構的過程質量和效果是可控的?結合對重構過程的理解,我們采用了Python腳本來支持第二步的工作。

2.3 使用Python支持重構過程

要規避以人工方式為主的重構過程,引入錯誤的風險,提升重構過程的質量及效率。需要引入Python腳本實現自動化工具支持重構過程的工作。下面以重構的關鍵步驟,自動化工具的應用目標進行列舉。

1、在熟悉業務及技術現狀階段,可以使用自動化工具對工程中現有的代碼、技術架構進行分析,獲取當前需要重構的代碼的依賴和調用關系信息,確定重構過程的變動影響,使用自動化的方式會更加的精準。

2、在確定重構方案階段,可以基于自動化工具產生的數據,支持重構方案的決策,包括是否需要重構,如何重構,調用方如何適配等。

3、在分階段實施階段,可以使用自動化的方式支持代碼的重構工作,包括需要重構的模塊的升級、調用方代碼的適配等。對比IDE提供的查找、替換等基礎工具,自動化工具可以批量處理更加復雜的重構工作。同時實施的階段通常是繁瑣且容易出錯的,但使用自動化的方式可以自動完成這些任務,并減少人為錯誤。

4、在效果評估及監控階段,可以使用自動化的方式對重構前后的代碼進行對比測試保證功能的一致性,收集關鍵指標數據,發現指標的異常。

03、用Python腳本實現模塊的調用關析分析

在實際的配置數據項的調用關系來看,公開的數據項可為幾種情況,對應的重構方案可有不同。

1、配置數據項僅在XXXSetting模塊內使用,這部分數據項不需要接入數據通路。

2、配置數據項在XXXSetting模塊內使用,也在其它的模塊中使用,這類數據項在XXXSetting模塊中維護,數據項需要接入數據通路。

3、配置數據項在XXXSetting模塊內沒有使用,只在一個模塊中使用,這類數據項應該遷移到使用該數據項的模塊中。

4、配置數據項在XXXSetting模塊內沒有使用,但在一個以上模塊中使用,這類數據項可以在XXXSetting模塊中維護,但數據項需要接入數據通路。

基于這樣的改造,XXXSetting模塊的數據項接口就可以全部不公開,對于配置數據項的變更,只影響依賴配置數據項的模塊。那么每個數據項的調用應該是如何重構呢,用手動查找及分析的方式成本過高,在項目實際過程評估及修改出錯的概率也會增高,我們使用Python腳本實現了調用關系的分析工具,為重構工作提前進行數據支持及決策。

3.1 提取公開數據項及類型

在分析數據項的外部調用情況之前,需要先提取XXXSetting類中所有公開的數據項。

3.1.1 公開數據項在OC類中的寫法

Setting文件由OC語言開發,在Setting頭文件件中公開的數據項的定義,OC類中成員變量的定義,書寫方式如下

@property (nonatomic, assign) BOOL value;
@property (nonatomic, copy) NSString *value1

3.1.2 提取的是變量類型和變量的名稱

因頭文件中,包含其它非成員變量的代碼,比如include、前置聲明、類定義、空代碼行、注釋、函數等,需要預處理下代碼及使用正則表達式變量定義代碼段,依次的讀取.h文件中的每一行代碼,以相關實現及的關鍵代碼如下。

去除注釋

因代碼中的注釋寫法存在不確定性,會對后面的正則匹配產生影響,故先把注釋刪除。

# 原代碼行 @property (nonatomic, copy) NSString *value1; // 注釋 ; * () 這些字符都有可能有,會影響后面的正則判斷
newline = re.sub(r'//.+', "", line)
# 處理過后的代碼行 @property (nonatomic, copy) NSString *value1;
提取數據項類型及數據項

去除注釋代碼之后,下一步為提取成員變量名稱及類型,可以使用正則中的分組匹配的能力,提取變量類型及變量名。這里使用了正則的原因是代碼的寫法存在不確定性,@property的寫法也會因變量類型不同而變化,故通過分組匹配的方式來實現。

# 原代碼行 @property (nonatomic, copy) NSString *value1;
matchObj = re.match(r"@property.+\)\s+(.*)", line, re.M|re.I)
if matchObj:
    # matchObj.group(1) 是成員變量類型和變量名 -- NSString *value1;

去除無用字符

這時的代碼行,因為寫法的不同及變量的不同,需要進行標準化,才能提取出變量類型及變量名,主要為去除 星號(*)。代碼行頭中的空格已經過濾(上行代碼中的\s+)。

# 原代碼行 NSString *value1;
newline = line.replace('*', '')
# 處理后的代碼行 NSString value1;

提取標準化后的數據項類型及數據項

這時代碼行中只剩下類型 空格 變量名 分號,使用正則的分組匹配,提取類型及變量名。

# 原代碼行 NSString value1;
# 正則表達式中\s匹配任何空白字符,包括空格、制表符、換頁符等等, 等價于[ \f\n\r\t\v],\s+代表一個或多個這類的字符
matchObj = re.match(r"(.*)\s+(.*);", line, re.M|re.I)
if matchObj:
    # valueType = NSString
    valueType = matchObj.group(1) 
     # valueName = value1  
    valueName = matchObj.group(2)

到這了一步,公開可訪問的數據項及類型的提取就已級完成,這時就可以轉換代碼,如果這時轉換代碼,會存在冗余,因為如果公開的變量在其它模塊中沒有使用,那實際上就不需要使用數據通路進行封裝,下一步應該分析調用關系之后,再進行。

3.2 數據項關聯調用組件

確定了公開的數據項之后,需要在工程源碼中查找每個數據項的調用點,之后再跟據調用點數據確定每個數據項在不同的組件中調用的情況。

數據項調用代碼常見于以下寫法,OC中也有其它的寫法,本文中以下寫法作為示例介紹調用關系的生成。

[XXXSetting share].value1

3.2.1. 查找每個數據項在文件中的調用

  • 原始數據項調用字串使用數據通路的數據項綁定。
  • 整體的思路為,依次的從每個文件中,全字匹配字符串,查找到一次,算作調用一次,保存到字典中,統一輸出到表格中。
# 定義個全局字典,存放每個數據項在不同的文件中調用的次數
# {數據項:{文件名:該文件內數據項調用的次數}}
valueCallInfoDic = {}


# 使用上節中,提取出來的數據項名,拼裝為實際調和時的寫法
realValueName = '[XXXSetting share].' + valueName


# fileNameList 為所有源碼文件(.m 和 .mm)
for fileName in fileNameList:
    # 記錄該文件調用數據項的次數
    callNum = 0
    # 記錄文件每個文件調用該數據項的次數信息
    fileCallInfoDic = {}
    # 依次的讀取源文件的每一行,匹配調用情況,記錄調用次數,及文件名,line 為代碼行
    for line in f:
        # 使用正則全字匹配,查找替換
        regAbKey = realValueName.replace('[', '\[')
        regAbKey = regAbKey.replace(']', '\]')
        regAbKey = regAbKey.replace('.', '\.')
        # pattern = \[XXXSetting share\]\.value1\b  主要為了防止數據項名有子串的情況
        pattern = r'' + fromstr + r'\b'
        matchObj = re.match(r'.*' + regAbKey +'', line, re.M|re.I)
        if matchObj:
            callNum = callNum + 1 
    if callNum > 0
        fileCallInfoDic[fileName] = str(callNum)
    # 如果有調用關系,則存儲
    if len(fileCallInfoDic) 
        valueCallInfoDic[valueName] = fileCallInfoDic

3.3 輸出為excel表格文件

使用Python分析的數據還是以機器語言的形式表式,需要以人類語言描述,將數據輸出為excel表格,這樣就可以借助于表格工具進行數據的查看及分析。

3.3.1 數據項的詳細使用情況輸出

表格的輸出Python沒有使用有excel操作的相關庫,使用 ,(逗號)作為分隔符,存儲為.csv文件,在excel中導入csv文件使用。

具體的實現為依次的將每個數據項的使用的組件,使用的文件及在這個文件文件中使用次數,輸出到.csv文件中。

# 表頭分別為,數據項,使用的組件,使用的文件,文件中使用次數
outfiledata = 'value , uselib , usefile , usenum\n'
# 遍歷全局字典valueCallInfoDic,獲取每個數據項 及數據項的調用信息
# {數據項:{文件名:該文件內數據項調用的次數}}
for.valueName , valueInfo in valueCallInfoDic.items():
    # 從數據項的調用信息中獲取,文件名和該文件內數據項調用的次數
    # {文件名:該文件內數據項調用的次數}
    for.fileName , callNum in valueCallInfoDic.items():
        outfiledata += valueName + " , "
        # libByFile 函數,實現根據文件獲取所在的組件名
        outfiledata += libByFile(fileName) + " , "
        outfiledata += fileName + " , "
        outfiledata += callNum + " \n"
表格數據示例

基于輸出的表格數據,可以比較容易的判斷每個數據項的優化影響范圍,下表為表格數據的示例。

△注:表格數據非真實業務場景數據△注:表格數據非真實業務場景數據

3.3.2 數據項的預分析統計輸出

基于數據的調用關系數據,確定每個數據項被每個組件使用的情況,并確定重構的方式。

同樣,表格的輸出Python沒有使用有excel操作的相關庫,使用 ,(逗號)作為分隔符,存儲為.csv文件,在excel中導入csv文件使用。

具體的實現為依次的讀取數據項,計算每個數據項被組件的使用情況,并將結果輸出到.csv文件中。

# 表頭分別為 ,數據項 ,使用的組件 ,組件中總使用次數 , 使用類型
outfiledata = 'value , uselib , usenum , usetype \n'
# 遍歷全局字典,獲取每個數據項 及數據項的調用信息
# {數據項:{文件名:該文件內數據項調用的次數}}
for.valueName , valueInfo in valueCallInfoDic.items():
    libCallInfo = {}
    # 從數據項的調用信息中獲取,文件名和該文件內數據項調用的次數
    # {文件名:該文件內數據項調用的次數}
    for.fileName , callNum in valueCallInfoDic.items():
        # libByFile 函數,實現根據文件獲取所在的組件名
        libName =  libByFile(fileName)
        if libName in libCallInfo:
            libCallInfo[libName] = int(libCallInfo[libName]) + int(callNum)
        else:
            libCallInfo[libName] = callNum
    # 每個組件的使用XXXSetting 的數據項情況
    hasSelfCall = False
    useType = ""
    for.libName in libCallInfo:
        if libName == "XXXSetting":
            hasSelfCall = True
            break
    if len(libCallInfo) == 1:
        if hasSelfCall:
            # 配置數據項僅在XXXSetting模塊內使用,這部分數據項不需要接入數據通路。
            useType = "selfCall"
        else:
            # 配置數據項在XXXSetting模塊內沒有使用,只在一個模塊中使用,這類數據項應該遷移到使用該數據項的模塊中。
            useType = "otherCall"
    else:
        if hasSelfCall:
            # 配置數據項在XXXSetting模塊內使用,也在其它的模塊中使用,這類數據項在XXXSetting模塊中維護,數據項需要接入數據通路。
            useType = "selfAndOtherCall"
        else:
            # 配置數據項在XXXSetting模塊內沒有使用,但在一個以上模塊中使用,這類數據項可以在XXXSetting模塊中維護,但數據項需要接入數據通路。
            useType = "othersCall"
    for.libName , libCallNum in libCallInfo.items():
        outfiledata += valueName + " , "
        outfiledata += libName + " , "
        outfiledata += libCallNum + " \n"
表格數據示例

基于輸出的表格數據,可以比較容易的判斷每個數據項應該如何整改,下表為表格數據的示例。

注:表格數據非真實業務場景數據注:表格數據非真實業務場景數據

04小結

以上的內容,介紹了代碼重構過程的工作及挑戰,同時以Python腳本實現分析模塊的調用關系的統計,基于該腳本,在重構工作開始之前,可以精確統計每個XXXSetting類對外公開的類成員屬性,被其它組件使用的情況。基于統計的數據,可以感知對應的每個成員屬性在App中的使用情況,且可容易的評估XXXSetting數據項重構升級為數據通路工作所帶來的影響。

當這部分工作,使用人工的方式實現,依次查找每個成員屬性的在App中的使用情況及分類記錄,是一件重復性高,出錯概率高的工作。而使用自動化工具,很好的規避了這些問題,且長期可積累。

責任編輯:武曉燕 來源: 百度Geek說
相關推薦

2022-07-04 07:37:51

模板模式重構

2021-05-26 08:50:37

JavaScript代碼重構函數

2017-07-13 11:08:52

PythonC模塊性能分析

2023-06-28 08:12:49

Python代碼重構

2017-08-08 16:07:57

Android 模塊化架構

2017-08-11 16:10:36

微信Android實踐

2010-03-26 15:55:47

Python腳本

2023-10-19 08:00:00

2010-03-25 16:31:55

Python代碼

2017-05-12 09:24:21

Python代碼Logger

2022-02-21 11:24:14

代碼工具開發

2024-02-20 22:13:48

Python項目Java

2022-08-08 08:10:42

Antlr解釋器工具

2021-08-03 08:13:48

重構API代碼

2024-02-22 10:27:00

Python開發

2012-07-27 10:30:12

重構

2023-07-26 06:43:07

函數調用

2024-04-26 13:21:32

可視化拖拽模塊

2011-06-09 15:27:01

JavaScript

2019-04-03 08:10:17

代碼架構信息
點贊
收藏

51CTO技術棧公眾號

91精品久久久久久久久| 欧美mv和日韩mv国产网站| 亚洲国产综合自拍| 91麻豆国产在线| 在线欧美亚洲| 国产午夜精品一区二区三区| 1314成人网| sese综合| 一区二区三区在线看| 久久精品人成| jizz中国少妇| 日韩成人一级片| 欧美黄色www| 男人的天堂官网| 国产成人tv| 69av一区二区三区| 国产女女做受ⅹxx高潮| 视频在线这里都是精品| 国产视频一区不卡| 成人91免费视频| 高潮毛片又色又爽免费| 国产精品大片免费观看| 日韩在线视频网站| 国产sm调教视频| 亚洲最好看的视频| 精品成人免费观看| 少妇愉情理伦片bd| 国产成人免费| 欧美综合一区二区三区| 97成人在线观看视频| 国产丝袜在线播放| 91看片淫黄大片一级在线观看| 成人欧美一区二区三区黑人孕妇| 高潮毛片又色又爽免费| 国产亚洲在线观看| 韩剧1988在线观看免费完整版| 国产盗摄x88av| 精品欧美激情在线观看| 亚洲欧美精品一区二区| 成年人在线观看av| 日韩超碰人人爽人人做人人添| 精品久久一区二区| 先锋资源在线视频| 久久精品九色| 欧美一区二区久久| 天堂在线精品视频| 国产一区二区视频在线看| 欧美久久久一区| av噜噜在线观看| 欧美成人免费全部网站| 欧美日韩一区精品| 欧美美女一级片| 亚洲伦理网站| 欧美一区二区三区在线电影| 欧美色图校园春色| 日韩高清一区| 日韩欧美在线不卡| 91成人在线观看喷潮蘑菇| 在这里有精品| 亚洲国产成人久久| 爱爱的免费视频| 视频一区中文| 日韩在线视频二区| 极品颜值美女露脸啪啪| 欧美精品一线| 韩国v欧美v日本v亚洲| 天天操天天干视频| 丝袜脚交一区二区| 成人黄色免费看| 不卡av中文字幕| 成人动漫一区二区在线| 久久久久久一区| av一本在线| 亚洲乱码精品一二三四区日韩在线| 欧美黑人在线观看| 漫画在线观看av| 欧美日韩一区二区三区视频| 天天久久综合网| 精品国产乱子伦一区二区| 亚洲欧美在线播放| 天天天天天天天天操| 影院欧美亚洲| 国产精品久久久久久久久久| 99久久精品国产色欲| www.66久久| 夜夜春亚洲嫩草影视日日摸夜夜添夜| 在线中文字幕第一页| 欧美日韩激情美女| 中文字幕成人在线视频| a级日韩大片| 一级做a爰片久久毛片美女图片| 天堂网avav| 久久精品一区二区国产| 91在线网站视频| 日韩精品系列| 亚洲精品国久久99热| 人妻内射一区二区在线视频| 国产成人免费av一区二区午夜| 亚洲第一精品夜夜躁人人爽 | 99re成人精品视频| 亚洲三级一区| 在线观看爽视频| 日韩一卡二卡三卡| 国产精品成人无码免费| 国产一区二区三区自拍| 国产精品久久91| 欧美一级做性受免费大片免费| 中文字幕免费不卡| 亚洲不卡中文字幕无码| 91麻豆精品| 国产一区二区欧美日韩| 日本熟妇成熟毛茸茸| 经典三级在线一区| 色女人综合av| 在线免费看h| 精品少妇一区二区三区在线视频| 日本激情视频一区二区三区| 久久久久国产精品午夜一区| 国产呦系列欧美呦日韩呦| 国产网站在线免费观看| 日本精品一区二区三区高清| 污片免费在线观看| 午夜精品婷婷| 91在线视频免费| 在线视频婷婷| 色婷婷亚洲婷婷| 国产精品伦子伦| 悠悠资源网久久精品| 99超碰麻豆| av在线影院| 这里只有精品视频在线观看| 亚洲综合第一区| 三级不卡在线观看| 日韩国产高清一区| 香蕉成人影院| 亚洲网站视频福利| 日韩在线播放中文字幕| 波多野洁衣一区| 成人性生活视频免费看| 欧美高清日韩| 国产精品自产拍高潮在线观看| 欧美精品久久久久久久久久丰满| 黄色精品在线看| 性囗交免费视频观看| 激情欧美国产欧美| 高清av免费一区中文字幕| 在线观看的网站你懂的| 日韩一区和二区| 我要看黄色一级片| 国产精品综合网| 日本a在线天堂| 国产精品极品| 26uuu久久噜噜噜噜| 深夜福利视频一区| 日本道色综合久久| 国产调教在线观看| 精品一区二区三区在线播放视频| 中文字幕日韩一区二区三区 | 欧美午夜一区二区三区免费大片| 免费黄色片网站| 美女精品自拍一二三四| 亚洲天堂av免费在线观看| 伊人久久精品| 欧美精品18videosex性欧美| 日本wwwxxxx| 色网站国产精品| 99成人在线观看| 国产91精品一区二区麻豆网站| 国产原创popny丨九色 | 一区二区三区天堂av| 一级片一区二区三区| 一区二区三区四区激情| 在线观看国产免费视频| 视频一区视频二区在线观看| 亚洲精品高清国产一线久久| 视频在线亚洲| 91爱视频在线| 免费大片黄在线| 亚洲国产精品热久久| 波多野结衣在线观看一区| 最新高清无码专区| 真人bbbbbbbbb毛片| 蜜臀久久99精品久久久久宅男| 视色,视色影院,视色影库,视色网| 巨人精品**| 国产精品美乳一区二区免费| 一色桃子av在线| 亚洲欧美日韩网| www.av导航| 色综合久久综合| 卡通动漫亚洲综合| 久久久久国产免费免费| 亚洲欧美手机在线| 亚洲欧美日韩在线观看a三区| 一区二区国产日产| 色爱av综合网| 91在线网站视频| 韩国精品主播一区二区在线观看| 免费成人高清视频| 国产精品一区在线看| 日韩女优毛片在线| 国产精品xxxxxx| 亚洲一二三专区| 少妇太紧太爽又黄又硬又爽小说| 成人av在线一区二区三区| 另类小说色综合| 中文在线不卡| 永久免费网站视频在线观看| 经典一区二区| 午夜久久福利| 精品欧美一区二区在线观看视频| 99re久久| 欧美中文字幕视频| 国产偷倩在线播放| 久久综合电影一区| 不卡在线视频| 亚洲精品日韩丝袜精品| 刘亦菲毛片一区二区三区| 欧美日韩一区二区三区不卡| 日韩一区二区视频在线| 亚洲一区二区免费视频| 国产高清视频免费在线观看| 国产调教视频一区| 日本一区二区三区网站| 成人午夜电影网站| 欧美熟妇精品一区二区| 精品一区二区三区香蕉蜜桃| 人人干人人视频| 久久久www| 国产精品第12页| 奶水喷射视频一区| 337p粉嫩大胆噜噜噜鲁| 亚洲麻豆av| 在线视频91| 少妇精品久久久一区二区三区| 国产乱码精品一区二区三区卡 | 久久国产亚洲精品| 国产视频精品网| 这里视频有精品| 成人欧美一区二区| 日韩中文字幕无砖| 91麻豆蜜桃| 欧美成人一级| 成人黄色在线免费观看| 日韩成人视屏| 成人av电影免费| 国产+成+人+亚洲欧洲在线| 高清国产一区| 欧美日本乱大交xxxxx| 午夜免费精品视频| 一本色道精品久久一区二区三区| 97超碰在线视| 国产综合自拍| 野外做受又硬又粗又大视频√| 国产亚洲精品久久久久久牛牛| 久久久久久天堂| 一区二区三区四区国产精品| 精品无码人妻一区二区三区| 亚洲一二三四区| 日韩 欧美 综合| 色综合久久久久综合体| 国产成人精品一区二区色戒| 欧美日韩精品免费| 国产免费久久久| 精品久久久网站| 神马久久高清| 亚洲图片在区色| av网站导航在线观看免费| 欧美精品亚州精品| 女厕盗摄一区二区三区| 国产精品av电影| 91精品网站在线观看| 高清视频一区| 久久不见久久见中文字幕免费| 亚洲午夜精品久久| 亚洲天堂男人| 国产精品久久久久9999小说| 激情成人午夜视频| 中文字幕人妻一区| 国产亚洲成aⅴ人片在线观看| 艳妇荡乳欲伦69影片| 五月天精品一区二区三区| 免费黄色片视频| 日韩午夜精品电影| 国产尤物视频在线| 欧美成人合集magnet| 亚洲欧美韩国| 97夜夜澡人人双人人人喊| 美女久久99 | 久久亚洲国产| 妺妺窝人体色777777| 国家队第一季免费高清在线观看| 国产目拍亚洲精品99久久精品| 日韩影院一区二区| 色呦呦国产精品| 成人h动漫精品一区二区无码| 亚洲人a成www在线影院| 麻豆福利在线观看| 国产精自产拍久久久久久| 好吊妞视频这里有精品 | 亚洲影视九九影院在线观看| 日本一道高清一区二区三区| 免费看av软件| 日韩精品成人一区二区三区| 五月天丁香社区| 综合久久综合久久| 最新国产中文字幕| 日韩av在线网页| 另类视频在线| 成人欧美一区二区三区黑人| 精品久久成人| 欧美性久久久久| 99久久99久久久精品齐齐| 91日韩中文字幕| 欧美日韩午夜在线视频| 青青青手机在线视频观看| 欧美国产日韩视频| 欧美成人一级| 超碰在线免费观看97| 日本不卡免费在线视频| 亚洲网站在线免费观看| 亚洲国产欧美精品| caoporn97在线视频| 国产精品视频yy9099| 国产精品一在线观看| 天天夜碰日日摸日日澡性色av| 国产成人免费网站| 福利所第一导航| 91麻豆精品国产91久久久 | 精品久久亚洲| 亚洲高清视频在线观看| 另类亚洲自拍| 无码人妻精品一区二区三区温州| 亚洲成人av一区二区三区| 午夜久久久久久噜噜噜噜| 久久久av一区| 日韩高清二区| av在线com| 成人午夜在线免费| 免费在线看黄网址| 精品国产乱码久久久久久牛牛| 五月天激情在线| 成人av免费看| 日韩视频在线一区二区三区| 老熟妇精品一区二区三区| 五月天亚洲精品| 全部免费毛片在线播放网站| 国产成人精品综合| 青青草国产免费一区二区下载| 五月婷婷六月丁香激情| 日韩毛片一二三区| 国产肥老妇视频| 久久久久久久久久久久久久久久久久av| 亚洲欧美日本国产| 国产探花在线精品| 手机在线观看国产精品| 久久夜色精品| 精品熟妇无码av免费久久| 欧美日韩亚洲高清一区二区| 秋霞午夜在线观看| 亚洲a成v人在线观看| 黄色日韩在线| www.久久国产| 欧美老年两性高潮| 在线观看小视频| 久久av二区| 蜜桃av噜噜一区| 九九免费精品视频| 日韩精品视频观看| 成人亚洲免费| 欧美黄网在线观看| 久久这里只有精品视频网| 亚洲一区二区天堂| 欧美精品久久久久久久久久| 亚洲ab电影| 伊人五月天婷婷| 午夜影院在线观看欧美| 国产乱子伦三级在线播放| 91传媒视频在线观看| 亚洲欧美高清| 久久国产美女视频| 精品视频—区二区三区免费| 激情中国色综合| 国产va亚洲va在线va| 国产日韩欧美精品一区| a网站在线观看| 国产成人精品久久亚洲高清不卡| 婷婷色综合网| 爱爱的免费视频| 日韩精品中文字幕一区| 最近高清中文在线字幕在线观看1| 椎名由奈jux491在线播放 | 91精品国产一区二区三区| 成人国产精品久久久网站| 91久久精品国产91性色tv| 粗大黑人巨茎大战欧美成人| 麻豆一区区三区四区产品精品蜜桃| 九色|91porny| 无码人妻丰满熟妇精品区| 欧美肥臀大乳一区二区免费视频|