日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

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

编程问答

走在技术前沿的 iOS 架构实现

發布時間:2025/3/18 编程问答 25 豆豆
生活随笔 收集整理的這篇文章主要介紹了 走在技术前沿的 iOS 架构实现 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

基于 Objective-C 實現的框架設計,YTKNetwork網絡層 + AOP替代基類 + MVVM + ReactiveObjC + JLRoutes路由

我理解的框架,就好比計算機的主板,房屋的建筑骨架,道路的基礎設施配套,框架搭的好,能直接影響開發者的開發心情,更能讓項目健壯性和擴展性大大增強。


? 要求

  • iOS 8.0+
  • Xcode 8.0+
  • Objective-C

? 測試 UI 什么樣子?

1.展示頁2.展示頁3.展示頁4.說明頁
登錄視圖示例展示跳轉頁面介紹頁面

? 安裝方法

安裝

在 iOS, 你需要在 Podfile 中添加.

source 'https://github.com/CocoaPods/Specs.git' platform :ios, '9.0' use_frameworks!# 提示組件框架pod 'SVProgressHUD', '~> 2.2.2'# 網絡請求框架pod 'YTKNetwork', '~> 2.0.3'# AOP面向切面pod 'Aspects', '~> 1.4.1'# 響應函數式框架pod 'ReactiveObjC', '~> 3.0.0'# 路由組件化解耦pod 'JLRoutes', '~> 2.0.5'# 提示組件框架pod 'SVProgressHUD', '~> 2.2.2'# 自動布局pod 'Masonry', '~> 1.0.2' 復制代碼

? 框架介紹

1.AOP 模式(Aspects-RunTime 代替基類)+ Category 方法交換

采用AOP思想,使用 Aspects 來完成替換 Controller ,View,ViewModel基類,和基類說拜拜

Casa反革命工程師 iOS應用架構談 view層的組織和調用方案 博客中提到一個疑問 是否有必要讓業務方統一派生ViewController

Casa大神回答是NO,原因如下

  • 使用派生比不使用派生更容易增加業務方的使用成本
  • 不使用派生手段一樣也能達到統一設置的目的 對于第一點,從 集成成本 ,上手成本 ,架構維護成本等因素入手,大神博客中也已經很詳細。
  • 框架不需要通過繼承即能夠對ViewController進行統一配置。業務即使脫離環境,也能夠跑完代碼,ViewController一旦放入框架環境,不需要添加額外的或者只需添加少量代碼,框架也能夠起到相應的作用 對于本人來說 ,具備這點的吸引力,已經足夠讓我有嘗試一番的心思了。

    對于OC來說,方法攔截很容易就想到自帶的黑魔法方法調配 Method Swizzling, 至于為ViewController做動態配置,自然非Category莫屬了 Method Swizzling 業界已經有非常成熟的三方庫 Aspects, 所以Demo代碼采用 Aspects 做方法攔截。

    + (void)load {[super load];[FKViewControllerIntercepter sharedInstance]; } // .... 單例初始化代碼- (instancetype)init {self = [super init];if (self) {/* 方法攔截 */// 攔截 viewDidLoad 方法[UIViewController aspect_hookSelector:@selector(viewDidLoad) withOptions:AspectPositionAfter usingBlock:^(id<AspectInfo>aspectInfo){[self _viewDidLoad:aspectInfo.instance];} error:nil];// 攔截 viewWillAppear:[UIViewController aspect_hookSelector:@selector(viewWillAppear:) withOptions:AspectPositionAfter usingBlock:^(id<AspectInfo> aspectInfo, BOOL animated){[self _viewWillAppear:animated controller:aspectInfo.instance];} error:NULL];}return self; } 復制代碼

    至于 Category 已經非常熟悉了

    @interface UIViewController (NonBase)/**去Model&&表征化參數列表*/ @property (nonatomic, strong) NSDictionary *params;/**ViewModel 屬性*/ @property (nonatomic, strong) id <FKViewControllerProtocol> viewModel;#pragma mark - 通用類/**返回Controller的當前bounds@param hasNav 是否有導航欄@param hasTabBar 是否有tabbar@return 坐標*/ - (CGRect)fk_visibleBoundsShowNav:(BOOL)hasNav showTabBar:(BOOL)hasTabBar;/**隱藏鍵盤*/ - (void)fk_hideKeyBoard; @end 復制代碼

    至此,我們已經實現了不繼承基類來實現對ViewController的配置,項目中的 View ViewModel 去基類原理如出一轍。

    2.View層采用 MVVM 設計模式,使用 ReactiveObjC 進行數據綁定

    -MVC-

    作為老牌思想MVC,大家早已耳熟能詳,MVC素有 Massive VC之稱,隨著業務增加,Controller將會越來越復雜,最終Controller會變成一個"神類", 即有網絡請求等代碼,又充斥著大量業務邏輯,所以為Controller減負,在某些情況下變得勢在必行

    -MVVM-

    MVVM是基于胖Model的架構思路建立的,然后在胖Model中拆出兩部分:Model和ViewModel (注:胖Model 是指包含了一些弱業務邏輯的Model) 胖Model實際上是為了減負 Controller 而存在的,而 MVVM 是為了拆分胖Model , 最終目的都是為了減負Controller。

    我們知道,蘋果MVC并沒有專門為網絡層代碼分專門的層級,按照以往習慣,大家都寫在了Controller 中,這也是Controller 變Massive得元兇之一,現在我們可以將網絡請求等諸如此類的代碼放到ViewModel中了 (文章后半部分將會描述ViewModel中的網絡請求)

    -數據流向-

    正常的網絡請求獲取數據,然后更新View自然不必多說,那么如果View產生了數據要怎么把數據給到Model,由于View不直接持有ViewModel,所以我們需要有個橋梁 ReactiveCocoa, 通過 Signal 來和 ViewModel 通信,這個過程我們使用 通知 或者 Target-Action也可以實現相同的效果,只不過沒有 ReactiveCocoa 如此方便罷了

    /* View -> ViewModel 傳遞數據示例 */ #pragma mark - Bind ViewModel - (void)bindViewModel:(id<FKViewModelProtocol>)viewModel withParams:(NSDictionary *)params {if ([viewModel isKindOfClass:[FKLoginViewModel class]]){FKLoginViewModel *_viewModel = (FKLoginViewModel *)viewModel;// 綁定賬號 View -> ViewModel 傳遞數據 @weakify(self);RAC(_viewModel, userAccount) = [[self.inputTextFiled.rac_textSignal takeUntil:self.rac_prepareForReuseSignal] map:^id _Nullable(NSString * _Nullable account) {@strongify(self);// 限制賬號長度if (account.length > 25) {self.inputTextFiled.text = [account substringToIndex:25];}return self.inputTextFiled.text;}];} } 復制代碼

    上面代碼給出了 View -> ViewModel 綁定的一個例子 具體一些詳情,可以直接看Demo MVVM一些總結:

  • View <-> C <-> ViewModel <-> Model 實際上應該稱之為MVCVM
  • Controller 將不再直接和 Model 進行綁定,而通過橋梁ViewModel
  • 最終 Controller 的作用變成一些UI的處理邏輯,和進行View和ViewModel的綁定
  • MVVM 和 MVC 兼容
  • 由于多了一層 ViewModel, 會需要寫一些膠水代碼,所以代碼量會增加
  • 3.網絡層使用 YTKNetwork 配合 ReactiveCocoa 封裝網絡請求,解決如何交付數據,交付什么樣的數據(去Model化)等問題

    YTKNetwork 是猿題庫 iOS 研發團隊基于 AFNetworking 封裝的 iOS 網絡庫,其實現了一套 High Level 的 API,提供了更高層次的網絡訪問抽象。

    筆者對 YTKNetwork 進行了一些封裝,結合 ReactiveCocoa,并提供 reFormatter 接口對服務器響應數據重新處理,靈活交付給業務層。 接下來,本文會回答兩個問題

  • 以什么方式將數據交付給業務層?
  • 交付什么樣的數據 ? 對于第一個問題
  • 以什么方式將數據交付給業務層?

    雖然 iOS應用架構談 網絡層設計方案 中 Casa大神寫到 盡量不要用block,應該使用代理 的確,Block難以追蹤和定位錯誤,容易內存泄漏, YTKNetwork 也提供代理方式回調

    @protocol YTKRequestDelegate <NSObject>@optional /// Tell the delegate that the request has finished successfully. /// /// @param request The corresponding request. - (void)requestFinished:(__kindof YTKBaseRequest *)request;/// Tell the delegate that the request has failed. /// /// @param request The corresponding request. - (void)requestFailed:(__kindof YTKBaseRequest *)request;@end 復制代碼

    前文有說過,MVVM 并不等于 ReactiveCocoa , 但是想要體驗最純正的 ReactiveCocoa 還是Block較為酸爽,Demo中筆者兩者都給出了代碼, 大家可以自行選擇和斟酌哈 我們看一下 YTKNetwork 和 ReactiveCocoa 結合的代碼

    - (RACSignal *)rac_requestSignal {[self stop];RACSignal *signal = [[RACSignal createSignal:^RACDisposable * _Nullable(id<RACSubscriber> _Nonnull subscriber) {// 請求起飛[self startWithCompletionBlockWithSuccess:^(__kindof YTKBaseRequest * _Nonnull request) {// 成功回調[subscriber sendNext:[request responseJSONObject]];[subscriber sendCompleted];} failure:^(__kindof YTKBaseRequest * _Nonnull request) {// 錯誤回調[subscriber sendError:[request error]];}];return [RACDisposable disposableWithBlock:^{// Signal銷毀 停止請求[self stop];}];}] takeUntil:[self rac_willDeallocSignal]];//設置名稱 便于調試if (DEBUG) {[signal setNameWithFormat:@"%@ -rac_xzwRequest", RACDescription(self)];}return signal; } 復制代碼

    寫了一個簡單的 Category FKBaseRequest+Rac.h ViewModel 中使用 RACCommand 封裝調用:

    - (RACCommand *)loginCommand {if (!_loginCommand) {@weakify(self);_loginCommand = [[RACCommand alloc] initWithSignalBlock:^RACSignal * _Nonnull(id _Nullable input) {@strongify(self);return [[[FKLoginRequest alloc] initWithUsr:self.userAccount pwd:self.password] rac_requestSignal];}];}return _loginCommand; } 復制代碼

    Block方式交付業務

    FKLoginRequest *loginRequest = [[FKLoginRequest alloc] initWithUsr:self.userAccount pwd:self.password]; return [[[loginRequest rac_requestSignal] doNext:^(id _Nullable x) {// 解析數據[[NSUserDefaults standardUserDefaults] setObject:@(YES) forKey:@"isLogin"];}] materialize]; 復制代碼

    Delegate方式交付業務

    FKLoginRequest *loginRequest = [[FKLoginRequest alloc] initWithUsr:self.userAccount pwd:self.password]; // 數據請求響應代理 通過代理回調 loginRequest.delegate = self; return [loginRequest rac_requestSignal];#pragma mark - YTKRequestDelegate - (void)requestFinished:(__kindof YTKBaseRequest *)request {// 解析數據[[NSUserDefaults standardUserDefaults] setObject:@(YES) forKey:@"isLogin"]; } 復制代碼

    交付什么樣的數據 ?

    現在諸如 JSONModel ,YYModel 之類的Json轉Model的庫也非常多,大多數Json對象,網絡請求成功直接就被轉成Model了 然而 iOS應用架構談 網絡層設計方案 中給出了兩種有意思的交付思路

  • 使用 reformer 對數據進行清洗
  • 去特定對象表征 (去Model)
  • Casa文章中好處已經寫得很詳細了,通過不同的 reformer 來重塑和交付不同的業務數據,可以說是非常靈活了

    使用 reformer 對數據進行清洗

    在網絡層封裝 FKBaseRequest.h 中 給出了 FKBaseRequestFeformDelegate 接口來重塑數據

    @protocol FKBaseRequestFeformDelegate <NSObject>/**自定義解析器解析響應參數@param request 當前請求@param jsonResponse 響應數據@return 自定reformat數據*/ - (id)request:(FKBaseRequest *)request reformJSONResponse:(id)jsonResponse;@end 然后在對應的 reformer 對數據進行重塑 #pragma mark - FKBaseRequestFeformDelegate - (id)request:(FKBaseRequest *)request reformJSONResponse:(id)jsonResponse {if([request isKindOfClass:FKLoginRequest.class]){// 在這里對json數據進行重新格式化}return jsonResponse; } 復制代碼

    也可以直接在子類的 RequestManager 中覆蓋父類方法達到一樣的效果

    /* FKLoginRequest.m */// 可以在這里對response 數據進行重新格式化, 也可以使用delegate 設置 reformattor - (id)reformJSONResponse:(id)jsonResponse { } 復制代碼

    去特定對象表征 (去Model)

    這思路可以說是業界的泥石流了 去Model也就是說,使用NSDictionary形式交付數據,對于網絡層而言,只需要保持住原始數據即可,不需要主動轉化成數據原型 但是會存在一些小問題

  • 去Model如何保持可讀性?
  • 復雜和多樣的數據結構如何解析?
  • Casa大神 提出了 使用EXTERN + Const 字符串形式,并建議字符串跟著reformer走,個人覺得很多時候API只需要一種解析格式,所以Demo跟著 APIManager 走,其他情況下常量字符串建議聽從 Casa大神 的建議, 常量定義:

    /* FKBaseRequest.h */ // 登錄token key FOUNDATION_EXTERN NSString *FKLoginAccessTokenKey;/* FKBaseRequest.m */ NSString *FKLoginAccessTokenKey = @"accessToken"; 復制代碼

    在 .h 和 .m 文件中要同時寫太多代碼,我們也可以使用局部常量的形式,只要在 .h 文件中定義即可

    // 也可以寫成 局部常量形式 static const NSString *FKLoginAccessTokenKey2 = @"accessToken"; 最終那么我們的reformer可能會變成這樣子 - (id)request:(FKBaseRequest *)request reformJSONResponse:(id)jsonResponse {if([request isKindOfClass:FKLoginRequest.class]){// 在這里對json數據進行重新格式化return @{FKLoginAccessTokenKey : jsonResponse[@"token"],};}return jsonResponse; } 復制代碼

    復雜和多樣的數據結構如何解析? 有時候,reformer 交付過來的數據,我們需要解析的可能是字符串類型,也可能是NSNumber類型,也有可能是數組 為此,筆者提供了一系列 Encode Decode方法,來降低解析的復雜度和安全性

    #pragma mark - Encode Decode 方法 // NSDictionary -> NSString FK_EXTERN NSString* DecodeObjectFromDic(NSDictionary *dic, NSString *key); // NSArray + index -> id FK_EXTERN id DecodeSafeObjectAtIndex(NSArray *arr, NSInteger index); // NSDictionary -> NSString FK_EXTERN NSString * DecodeStringFromDic(NSDictionary *dic, NSString *key); // NSDictionary -> NSString ? NSString : defaultStr FK_EXTERN NSString* DecodeDefaultStrFromDic(NSDictionary *dic, NSString *key,NSString * defaultStr); // NSDictionary -> NSNumber FK_EXTERN NSNumber * DecodeNumberFromDic(NSDictionary *dic, NSString *key); // NSDictionary -> NSDictionary FK_EXTERN NSDictionary *DecodeDicFromDic(NSDictionary *dic, NSString *key); // NSDictionary -> NSArray FK_EXTERN NSArray *DecodeArrayFromDic(NSDictionary *dic, NSString *key); FK_EXTERN NSArray *DecodeArrayFromDicUsingParseBlock(NSDictionary *dic, NSString *key, id(^parseBlock)(NSDictionary *innerDic));#pragma mark - Encode Decode 方法 // (nonull Key: nonull NSString) -> NSMutableDictionary FK_EXTERN void EncodeUnEmptyStrObjctToDic(NSMutableDictionary *dic,NSString *object, NSString *key); // nonull objec -> NSMutableArray FK_EXTERN void EncodeUnEmptyObjctToArray(NSMutableArray *arr,id object); // (nonull (Key ? key : defaultStr) : nonull Value) -> NSMutableDictionary FK_EXTERN void EncodeDefaultStrObjctToDic(NSMutableDictionary *dic,NSString *object, NSString *key,NSString * defaultStr); // (nonull Key: nonull object) -> NSMutableDictionary FK_EXTERN void EncodeUnEmptyObjctToDic(NSMutableDictionary *dic,NSObject *object, NSString *key); 復制代碼

    我們的reformer可以寫成這樣子

    #pragma mark - FKBaseRequestFeformDelegate - (id)request:(FKBaseRequest *)request reformJSONResponse:(id)jsonResponse {if([request isKindOfClass:FKLoginRequest.class]){// 在這里對json數據進行重新格式化return @{FKLoginAccessTokenKey : DecodeStringFromDic(jsonResponse, @"token")};}return jsonResponse; } 復制代碼

    解析有可能是這樣子

    NSString *token = DecodeStringFromDic(jsonResponse, FKLoginAccessTokenKey) 復制代碼

    好了,至此我們解決了兩個問題

  • 以什么方式將數據交付給業務層 答:delegate 最佳,block為次
  • 交付什么樣的數據 答:純字典,去Model
  • 4.采用 JLRoutes 路由對應用進行組件化解耦

    帶著問題思考如何才能設計出最好的組件化路由:

    • 1)3D-Touch功能或者點擊推送消息,要求外部跳轉到App內部一個很深層次的一個界面。
    • 2)自家的一系列App之間如何相互跳轉?
    • 3)如何解除App組件之間和App頁面之間的耦合性?
    • 4)如何能統一iOS和Android兩端的頁面跳轉邏輯?甚至如何能統一三端的請求資源的方式?
    • 5)如果使用了動態下發配置文件來配置App的跳轉邏輯,那么如果做到iOS和Android兩邊只要共用一套配置文件?
    • 6)如果App出現bug了,如何不用JSPatch,就能做到簡單的熱修復功能?
    • 7)如何在每個組件間調用和頁面跳轉時都進行埋點統計?每個跳轉的地方都手寫代碼埋點?利用Runtime AOP ?
    • 8)如何在每個組件間調用的過程中,加入調用的邏輯檢查,令牌機制,配合灰度進行風控邏輯?
    • 9)如何在App任何界面都可以調用同一個界面或者同一個組件?只能在AppDelegate里面注冊單例來實現?

    iOS應用架構談 組件化方案 一文中 Casa 針對 蘑菇街組件化 提出了質疑,質疑點主要在這幾方面

  • App啟動時組件需要注冊URL
  • URL調用組件方式不太好傳遞類似 UIImage 等非常規對象
  • URL需要添加額外參數可讀性差,所以沒必要使用URL
  • 對于 App啟動時組件需要注冊URL 顧慮主要在于,注冊的URL需要在應用生存周期內常駐內存,如果是注冊Class還好些,如果注冊的是實例,消耗的內存就非常可觀了

    #pragma mark - 路由表 NSString *const FKNavPushRoute = @"/com_madao_navPush/:viewController"; NSString *const FKNavPresentRoute = @"/com_madao_navPresent/:viewController"; NSString *const FKNavStoryBoardPushRoute = @"/com_madao_navStoryboardPush/:viewController"; NSString *const FKComponentsCallBackRoute = @"/com_madao_callBack/*"; 復制代碼

    而且JLRoutes 還支持 * 來進行通配,路由表如何編寫大家可以自由發揮 對應的路由事件 handler

    // push // 路由 /com_madao_navPush/:viewController [[JLRoutes globalRoutes] addRoute:FKNavPushRoute handler:^BOOL(NSDictionary<NSString *,id> * _Nonnull parameters) {dispatch_async(dispatch_get_main_queue(), ^{[self _handlerSceneWithPresent:NO parameters:parameters];});return YES; }];// present // 路由 /com_madao_navPresent/:viewController [[JLRoutes globalRoutes] addRoute:FKNavPresentRoute handler:^BOOL(NSDictionary<NSString *,id> * _Nonnull parameters) {dispatch_async(dispatch_get_main_queue(), ^{[self _handlerSceneWithPresent:YES parameters:parameters];});return YES; }];#pragma mark - Private /// 處理跳轉事件 - (void)_handlerSceneWithPresent:(BOOL)isPresent parameters:(NSDictionary *)parameters {// 當前控制器NSString *controllerName = [parameters objectForKey:FKControllerNameRouteParam];UIViewController *currentVC = [self _currentViewController];UIViewController *toVC = [[NSClassFromString(controllerName) alloc] init];toVC.params = parameters;if (currentVC && currentVC.navigationController) {if (isPresent) {[currentVC.navigationController presentViewController:toVC animated:YES completion:nil];}else{[currentVC.navigationController pushViewController:toVC animated:YES];}} } 復制代碼

    通過URL中傳入的組件名動態注冊,處理相應跳轉事件,并不需要每個組件一一注冊 使用URL路由,必然URL會散落到代碼各個地方

    NSString *key = @"key"; NSString *value = @"value"; NSString *url = [NSString stringWithFormat:@"/com_madao_navPush/%@?%@=%@", NSStringFromClass(ViewController.class), key, value]; [[UIApplication sharedApplication] openURL:[NSURL URLWithString:url]]; 復制代碼

    諸如此類丑陋的代碼,散落在各個地方的話簡直會讓人頭皮發麻, 所以筆者在 JLRoutes+GenerateURL.h 寫了一些 Helper方法

    /**避免 URL 散落各處, 集中生成URL@param pattern 匹配模式@param parameters 附帶參數@return URL字符串*/ + (NSString *)fk_generateURLWithPattern:(NSString *)pattern parameters:(NSArray *)parameters;/**避免 URL 散落各處, 集中生成URL額外參數將被 ?key=value&key2=value2 樣式給出@param pattern 匹配模式@param parameters 附加參數@param extraParameters 額外參數@return URL字符串*/ + (NSString *)fk_generateURLWithPattern:(NSString *)pattern parameters:(NSArray *)parameters extraParameters:(NSDictionary *)extraParameters;/**解析NSURL對象中的請求參數 http://madao?param1=value1?m2=value2 解析成 @{param1:value1, param2:value2}@param URL NSURL對象@return URL字符串*/ + (NSDictionary *)fk_parseParamsWithURL:(NSURL *)URL;/**將參數對象進行url編碼將@{param1:value1, param2:value2} 轉換成 ?param1=value1&param2=value2@param dic 參數對象@return URL字符串*/ + (NSString *)fk_mapDictionaryToURLQueryString:(NSDictionary *)dic; 復制代碼

    宏定義Helper

    #undef JLRGenRoute #define JLRGenRoute(Schema, path) \ ([NSString stringWithFormat: @"%@:/%@", \ Schema, \ path])#undef JLRGenRouteURL #define JLRGenRouteURL(Schema, path) \ ([NSURL URLWithString: \ JLRGenRoute(Schema, path)]) 復制代碼

    最終我們的調用可以變成

    NSString *router = [JLRoutes fk_generateURLWithPattern:FKNavPushRoute parameters:@[NSStringFromClass(ViewController.class)] extraParameters:nil]; [[UIApplication sharedApplication] openURL:JLRGenRouteURL(FKDefaultRouteSchema, router)]; 復制代碼

    ? 整理制作

    Casa Taloyum:https://casatwy.com/modulization_in_action.html

    簡書博客:http://www.jianshu.com/p/921dd65e79cb


    ? 聯系

    • 微信 : WhatsXie
    • 郵件 : ReverseScale@iCloud.com
    • 博客 : https://reversescale.github.io
    • 源碼 : https://github.com/ReverseScale/OCTemplate

    總結

    以上是生活随笔為你收集整理的走在技术前沿的 iOS 架构实现的全部內容,希望文章能夠幫你解決所遇到的問題。

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

    主站蜘蛛池模板: 97色爱| 免费黄色激情视频 | 96免费视频 | 亚洲AV无码成人精品区东京热 | 日本一级片在线播放 | 亚洲艹 | 91丨九色丨丰满人妖 | 日本少妇吞精囗交 | 中文字幕一区二区久久人妻网站 | 成人看| 国产精品福利一区二区三区 | 亚洲成a人片在线www | 中文字幕 欧美日韩 | 一边摸一边抽搐一进一出视频 | 欧美日韩在线视频免费播放 | 国产免费一区二区三区 | zjzjzjzjzj亚洲女人 | 视频免费观看在线 | 婷婷激情综合 | 香蕉蜜桃视频 | 91精品免费视频 | 大胸美女被爆操 | 欧美在线视频播放 | 成人久久一区二区 | 含羞草一区二区三区 | 一本在线免费视频 | 欧美999 | 成人深夜福利 | 捆绑束缚调教 | 天天婷婷 | 丰满少妇在线观看bd | 一区二区久久精品66国产精品 | 亚洲AV无码国产成人久久 | 五月情网 | 国产精品精品国产 | 欧洲性生活视频 | 国产成人精品影院 | 日本一本高清视频 | 奇米成人 | 蜜桃网av| 色性av| 在线视频观看一区 | 尹人香蕉| 亚洲一区二区在线电影 | 亚洲影院在线观看 | 我要看黄色大片 | 黄色最新网址 | 一级免费av | 亚洲免费大片 | 国产aaa毛片 | 无套内谢少妇毛片 | 超碰在线影院 | 疯狂揉花蒂控制高潮h | mm131在线 | 亚洲人午夜射精精品日韩 | 啪啪免费av | 天天干天天弄 | 欧美视频免费看 | 亚洲精品一区二区三区精华液 | 日韩欧美一区二区一幕 | 五月天在线播放 | 在线观看视频一区二区 | 国产麻豆a毛片 | 实拍澡堂美女洗澡av | 日韩视频免费在线播放 | 成人1区2区3区| 五月综合激情 | 天天人人精品 | 麻豆精品国产精华精华液好用吗 | 无码人妻精品中文字幕 | 中文在线字幕观看 | 一区二区传媒有限公司 | 亚洲精品国产精品国 | 香蕉在线影院 | 欧美日本亚洲韩国国产 | 日韩免费看 | 午夜福利啪啪片 | 香蕉av网站 | 日日艹夜夜艹 | 播播成人网 | 日韩一区二区三区精 | 日本精品视频一区二区三区 | 亚洲影视一区 | www.xxx.日本| 青娱乐最新官网 | 日韩不卡高清视频 | 欧美18一20男同69gay | 在线亚洲成人 | 丰满饥渴老女人hd | 日韩一区视频在线 | 亚洲AV成人无码久久 | 禁断介护av一区二区 | 91网页在线观看 | 色吊丝中文字幕 | 免费av小说 | 99在线无码精品入口 | 久久22| 最新日韩在线视频 | 无码人妻精品一区二区三区99不卡 |