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

歡迎訪問 生活随笔!

生活随笔

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

编程问答

iOS,Objective-C Runtime

發(fā)布時間:2023/12/31 编程问答 31 豆豆
生活随笔 收集整理的這篇文章主要介紹了 iOS,Objective-C Runtime 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

1.簡介

2.與Runtime交互

3.Runtime術語

4.消息

5.動態(tài)方法解析

6.消息轉發(fā)

7.健壯的實例變量(Non Fragile ivars)

8.Objective-C Associated Objects

9.Method Swizzling

10.總結

1.簡介

? ? 參考博客:http://yulingtianxia.com/blog/2014/11/05/objective-c-runtime/

? ? ? 因為Objc是一門動態(tài)語言,所以它總是想辦法把一些決定工作從編譯連接推遲到運行時。也就是說只有編譯器是不夠的,還需要一個運行時系統(tǒng) (runtime system) 來執(zhí)行編譯后的代碼。這就是 Objective-C Runtime 系統(tǒng)存在的意義,它是整個Objc運行框架的一塊基石。?Runtime基本是用C和匯編寫的,可見蘋果為了動態(tài)系統(tǒng)的高效而作出的努力。

? ? ? Runtime其實有兩個版本:“modern”和 “l(fā)egacy”。我們現(xiàn)在用的 Objective-C 2.0 采用的是現(xiàn)行(Modern)版的Runtime系統(tǒng),只能運行在 iOS 和 OS X 10.5 之后的64位程序中。而OS X較老的32位程序仍采用 Objective-C 1中的(早期)Legacy 版本的 Runtime 系統(tǒng)。這兩個版本最大的區(qū)別在于當你更改一個類的實例變量的布局時,在早期版本中你需要重新編譯它的子類,而現(xiàn)行版就不需要。

? ? ? 面向切面編程:(AOP是Aspect Oriented Program的首字母縮寫)這種在運行時,動態(tài)地將代碼切入到類的指定方法、指定位置上的編程思想就是面向切面的編程。

?

2.與Runtime交互

?

   ?Objc 從三種不同的層級上與 Runtime 系統(tǒng)進行交互,分別是通過 Objective-C 源代碼,通過 Foundation 框架的NSObject類定義的方法,通過對 runtime 函數(shù)的直接調用。

?

? ?2.1.Objective-C源代碼

?

? ? ? ? ? 大部分情況下你就只管寫你的Objc代碼就行,runtime 系統(tǒng)自動在幕后辛勤勞作著。消息的執(zhí)行會使用到一些編譯器為實現(xiàn)動態(tài)語言特性而創(chuàng)建的數(shù)據(jù)結構和函數(shù),Objc中的類、方法和協(xié)議等在 runtime 中都由一些數(shù)據(jù)結構來定義

?

? ?2.2.NSObject的方法

?

? ? ? ? ? Cocoa 中大多數(shù)類都繼承于NSObject類,也就自然繼承了它的方法。最特殊的例外是NSProxy,它是個抽象超類,它實現(xiàn)了一些消息轉發(fā)有關的方法,可以通過繼承它來實現(xiàn)一個其他類的替身類或是虛擬出一個不存在的類。

?

? ? ? ? ? 有的NSObject中的方法起到了抽象接口的作用,比如description方法需要你重載它并為你定義的類提供描述內容。NSObject還有些方法能在運行時獲得類的信息,并檢查一些特性,比如class返回對象的類;isKindOfClass:和isMemberOfClass:則檢查對象是否在指定的類繼承體系中;respondsToSelector:檢查對象能否響應指定的消息;conformsToProtocol:檢查對象是否實現(xiàn)了指定協(xié)議類的方法;methodForSelector:則返回指定方法實現(xiàn)的地址。

?

? ?2.3.Runtime的函數(shù)

? ? ? ? ? ?Runtime 系統(tǒng)是一個由一系列函數(shù)和數(shù)據(jù)結構組成,具有公共接口的動態(tài)共享庫。頭文件存放于/usr/include/objc目錄下。許多函數(shù)允許你用純C代碼來重復實現(xiàn) Objc 中同樣的功能。雖然有一些方法構成了NSObject類的基礎,但是你在寫 Objc 代碼時一般不會直接用到這些函數(shù)的,除非是寫一些 Objc 與其他語言的橋接或是底層的debug工作。?

3.Runtime術語

? objc_msgSend:方法吧,都會說它的偽代碼如下或類似的邏輯,反正就是獲取 IMP (函數(shù)指針,保存了方法地址)并調用,因為?objc_msgSend?是用匯編語言寫的,針對不同架構有不同的實現(xiàn)。它的真身是這樣的:

id objc_msgSend(id self, SEL _cmd, ...) {Class class = object_getClass(self);IMP imp = class_getMethodImplementation(class, _cmd);return imp ? imp(self, _cmd, ...) : 0;}

?

?

? ?3.1.SEL

? ? ? ? ?objc_msgSend函數(shù)第二個參數(shù)類型為SEL,它是selector在Objc中的表示類型(Swift中是Selector類)。selector是方法選擇器,可以理解為區(qū)分方法的 ID,而這個 ID 的數(shù)據(jù)結構是SEL:

?

? ? ? ? ??typedef struct objc_selector *SEL;

?

? ? ? ? ??其實它就是個映射到方法的C字符串,你可以用 Objc 編譯器命令@selector()或者 Runtime 系統(tǒng)的sel_registerName函數(shù)來獲得一個SEL類型的方法選擇器。

?

? ? ? ? ?不同類中相同名字的方法所對應的方法選擇器是相同的,即使方法名字相同而變量類型不同也會導致它們具有相同的方法選擇器,于是 Objc 中方法命名有時會帶上參數(shù)類型(NSNumber一堆抽象工廠方法)。

?

? ?3.2.id

   objc_msgSend第一個參數(shù)類型為id,大家對它都不陌生,它是一個指向類實例的指針:

? ? ? ? ??typedef struct objc_object *id;

   ?那objc_object又是啥呢:

?   struct objc_object { Class isa; };? ? ? ? ? ? ? ? ?

? ? ?  ?objc_object結構體包含一個isa指針,根據(jù)isa指針就可以順藤摸瓜找到對象所屬的類。

   ? ?PS:isa指針不總是指向實例對象所屬的類,不能依靠它來確定類型,而是應該用class方法來確定實例對象的類。因為KVO的實現(xiàn)機理就是將被觀察對象的isa指針指向一個中間類而不是真實的類,這是一種叫做 isa-swizzling 的技術,詳見官方文檔

 

?

? ?3.3.Class

?

    ?之所以說isa是指針是因為Class其實是一個指向objc_class結構體的指針:

?

? ? ? ? ? ?typedef struct objc_class *Class;

? ? ? ? ??objc_class里面的東西:

?

struct objc_class {Class isa OBJC_ISA_AVAILABILITY;#if !__OBJC2__Class super_class OBJC2_UNAVAILABLE;const char *name OBJC2_UNAVAILABLE;long version OBJC2_UNAVAILABLE;long info OBJC2_UNAVAILABLE;long instance_size OBJC2_UNAVAILABLE;struct objc_ivar_list *ivars OBJC2_UNAVAILABLE;struct objc_method_list **methodLists OBJC2_UNAVAILABLE;struct objc_cache *cache OBJC2_UNAVAILABLE;struct objc_protocol_list *protocols OBJC2_UNAVAILABLE;#endif} OBJC2_UNAVAILABLE;

?

?

?可以看到運行時一個類還關聯(lián)了它的超類指針,類名,成員變量,方法,緩存,還有附屬的協(xié)議。

?

PS:OBJC2_UNAVAILABLE之類的宏定義是蘋果在 Objc 中對系統(tǒng)運行版本進行約束的黑魔法,為的是兼容非Objective-C 2.0的遺留邏輯,但我們仍能從中獲得一些有價值的信息,有興趣的可以查看源代碼。

Objective-C 2.0 的頭文件雖然沒暴露出objc_class結構體更詳細的設計,我們依然可以從Objective-C 1.0 的定義中小窺端倪:

objc_class結構體中:ivarsobjc_ivar_list指針;methodLists是指向objc_method_list指針的指針。也就是說可以動態(tài)修改*methodLists的值來添加成員方法,這也是Category實現(xiàn)的原理,同樣解釋了Category不能添加屬性的原因。

其中objc_ivar_list和objc_method_list分別是成員變量列表和方法列表:

?

struct objc_ivar_list {int ivar_count OBJC2_UNAVAILABLE;#ifdef __LP64__int space OBJC2_UNAVAILABLE;#endif/* variable length structure */struct objc_ivar ivar_list[1] OBJC2_UNAVAILABLE;} OBJC2_UNAVAILABLE;struct objc_method_list {struct objc_method_list *obsolete OBJC2_UNAVAILABLE;int method_count OBJC2_UNAVAILABLE;#ifdef __LP64__int space OBJC2_UNAVAILABLE;#endif/* variable length structure */struct objc_method method_list[1] OBJC2_UNAVAILABLE;}

?

?

? ? ??如果你C語言不是特別好,可以直接理解為objc_ivar_list結構體存儲著objc_ivar數(shù)組列表,而objc_ivar結構體存儲了類的單個成員變量的信息;同理objc_method_list結構體存儲著objc_method數(shù)組列表,而objc_method結構體存儲了類的某個方法的信息。

? ? ??不知道你是否注意到了objc_class中也有一個isa對象,這是因為一個 ObjC 類本身同時也是一個對象,為了處理類和對象的關系,runtime 庫創(chuàng)建了一種叫做元類 (Meta Class) 的東西,類對象所屬類型就叫做元類,它用來表述類對象本身所具備的元數(shù)據(jù)。類方法就定義于此處,因為這些方法可以理解成類對象的實例方法。每個類僅有一個類對象,而每個類對象僅有一個與之相關的元類。當你發(fā)出一個類似[NSObject alloc]的消息時,你事實上是把這個消息發(fā)給了一個類對象 (Class Object) ,這個類對象必須是一個元類的實例,而這個元類同時也是一個根元類 (root meta class) 的實例。所有的元類最終都指向根元類為其超類。所有的元類的方法列表都有能夠響應消息的類方法。所以當?[NSObject alloc]?這條消息發(fā)給類對象的時候,objc_msgSend()會去它的元類里面去查找能夠響應消息的方法,如果找到了,然后對這個類對象執(zhí)行方法調用。

上圖實線是?super_class?指針,虛線是isa指針。 有趣的是根元類的超類是NSObject,而isa指向了自己,而NSObject的超類為nil,也就是它沒有超類。

? ? ? ? 3.3.1.Method

      Method是一種代表類中的某個方法的類型。

       ?typedef struct objc_method *Method;

objc_method在上面的方法列表中提到過,它存儲了方法名,方法類型和方法實現(xiàn):

struct objc_method {

? ? SEL method_name? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? OBJC2_UNAVAILABLE;

? ? char *method_types ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? OBJC2_UNAVAILABLE;

? ? IMP method_imp ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? OBJC2_UNAVAILABLE;

}? ?

  • 方法名類型為SEL,前面提到過相同名字的方法即使在不同類中定義,它們的方法選擇器也相同。
  • 方法類型method_types是個char指針,其實存儲著方法的參數(shù)類型和返回值類型。
  • method_imp指向了方法的實現(xiàn),本質上是一個函數(shù)指針,后面會詳細講到。

?

?

? ? ? ? 3.3.2.lvar

     ? ??Ivar是一種代表類中實例變量的類型。

       ?typedef struct objc_ivar *Ivar;

objc_ivar在上面的成員變量列表中也提到過:

struct objc_ivar {char *ivar_name OBJC2_UNAVAILABLE;char *ivar_type OBJC2_UNAVAILABLE;int ivar_offset OBJC2_UNAVAILABLE;#ifdef __LP64__int space OBJC2_UNAVAILABLE;#endif}

?

?可以根據(jù)實例查找其在類中的名字,也就是“反射”:

#import <objc/runtime.h>-(NSString *)nameWithInstance:(id)instance {unsigned int numIvars = 0;NSString *key=nil;Ivar * ivars = class_copyIvarList([self class], &numIvars);for(int i = 0; i < numIvars; i++) {Ivar thisIvar = ivars[i];const char *type = ivar_getTypeEncoding(thisIvar);NSString *stringType = [NSString stringWithCString:type encoding:NSUTF8StringEncoding];if (![stringType hasPrefix:@"@"]) {continue;}if ((object_getIvar(self, thisIvar) == instance)) {//此處若 crash 不要慌! key = [NSString stringWithUTF8String:ivar_getName(thisIvar)];break;}}free(ivars);return key;}

class_copyIvarList 函數(shù)獲取的不僅有實例變量,還有屬性。但會在原本的屬性名前加上一個下劃線。

遍歷屬性

//遍歷UIPageControl屬性unsigned int count = 0;Ivar *ivars = class_copyIvarList([UIPageControl class], &count);for (int i = 0; i < count; i++) {Ivar ivar = ivars[i];//獲取所有私有屬性const char *property = ivar_getName(ivar);NSLog(@"%@",[[NSString alloc]initWithCString:property encoding:NSUTF8StringEncoding]);}

?

?

? ?3.4.IMP

?

    ?IMPobjc.h中的定義是:

?

   typedef id (*IMP)(id, SEL, ...);?????????

?

? ? ? ? 它就是一個函數(shù)指針,這是由編譯器生成的。當你發(fā)起一個 ObjC 消息之后,最終它會執(zhí)行的那段代碼,就是由這個函數(shù)指針指定的。而?IMP?這個函數(shù)指針就指向了這個方法的實現(xiàn)。既然得到了執(zhí)行某個實例某個方法的入口,我們就可以繞開消息傳遞階段,直接執(zhí)行方法,這在后面會提到。?

你會發(fā)現(xiàn)IMP指向的方法與objc_msgSend函數(shù)類型相同,參數(shù)都包含idSEL類型。每個方法名都對應一個SEL類型的方法選擇器,而每個實例對象中的SEL對應的方法實現(xiàn)肯定是唯一的,通過一組idSEL參數(shù)就能確定唯一的方法實現(xiàn)地址;反之亦然。

?

?

? ?3.5.Cache

   ?在runtime.h中Cache的定義如下:

    typedef struct objc_cache *Cache

還記得之前objc_class結構體中有一個struct objc_cache *cache吧,它到底是緩存啥的呢,先看看objc_cache的實現(xiàn):

struct objc_cache {unsigned int mask /* total = mask + 1 */ OBJC2_UNAVAILABLE;unsigned int occupied OBJC2_UNAVAILABLE;Method buckets[1] OBJC2_UNAVAILABLE;};

Cache為方法調用的性能進行優(yōu)化,通俗地講,每當實例對象接收到一個消息時,它不會直接在isa指向的類的方法列表中遍歷查找能夠響應消息的方法,因為這樣效率太低了,而是優(yōu)先在Cache中查找。Runtime 系統(tǒng)會把被調用的方法存到Cache中(理論上講一個方法如果被調用,那么它有可能今后還會被調用),下次查找的時候效率更高。這根計算機組成原理中學過的 CPU 繞過主存先訪問Cache的道理挺像,蘋果為提高Cache命中率。

?

? ?3.6.Property?

property標記了類中的屬性,這個不必多說大家都很熟悉,它是一個指向objc_property結構體的指針:

?

typedef struct objc_property *Property;

?

typedef struct objc_property *objc_property_t;//這個更常用

?

可以通過class_copyPropertyList?和?protocol_copyPropertyList方法來獲取類和協(xié)議中的屬性:

?

objc_property_t *class_copyPropertyList(Class cls, unsigned int *outCount)

objc_property_t *protocol_copyPropertyList(Protocol *proto, unsigned int *outCount)

?

?返回類型為指向指針的指針,哈哈,因為屬性列表是個數(shù)組,每個元素內容都是一個objc_property_t指針,而這兩個函數(shù)返回的值是指向這個數(shù)組的指針。

?

舉個栗子,先聲明一個類:

@interface Lender : NSObject {float alone;}@property float alone;@end

?

你可以用下面的代碼獲取屬性列表:

id LenderClass = objc_getClass("Lender");
unsigned int outCount;
objc_property_t *properties = class_copyPropertyList(LenderClass, &outCount);

?

你可以用property_getName函數(shù)來查找屬性名稱:

const char *property_getName(objc_property_t property)

?

你可以用class_getProperty protocol_getProperty通過給出的名稱來在類和協(xié)議中獲取屬性的引用:

objc_property_t class_getProperty(Class cls, const char *name)

objc_property_t protocol_getProperty(Protocol *proto, const char *name, BOOL isRequiredProperty, BOOL isInstanceProperty)

?

你可以用property_getAttributes函數(shù)來發(fā)掘屬性的名稱和@encode類型字符串:

const char *property_getAttributes(objc_property_t property)

?

把上面的代碼放一起,你就能從一個類中獲取它的屬性啦:

#import <objc/runtime.h>#import "Lenaer.h"id LenderClass = objc_getClass("Lender");unsigned int outCount, i;objc_property_t *properties = class_copyPropertyList(LenderClass, &outCount);for (i = 0; i < outCount; i++) {objc_property_t property = properties[i];fprintf(stdout, "%s %s\n", property_getName(property), property_getAttributes(property));}?

對比下 class_copyIvarList 函數(shù),使用 class_copyPropertyList 函數(shù)只能獲取類的屬性,而不包含成員變量。但此時獲取的屬性名是不帶下劃線的。

?

?

4.消息

Objc 中發(fā)送消息是用中括號([])把接收者和消息括起來,而直到運行時才會把消息與方法實現(xiàn)綁定。

有關消息發(fā)送和消息轉發(fā)機制的原理,可以查看這篇文章。

? ?4.1.objc_msgSend函數(shù)

? ? ? ? ?編譯器會根據(jù)情況在objc_msgSend,?objc_msgSend_stret,?objc_msgSendSuper, 或?objc_msgSendSuper_stret四個方法中選擇一個來調用。如果消息是傳遞給超類,那么會調用名字帶有”Super”的函數(shù);如果消息返回值是數(shù)據(jù)結構而不是簡單值時,那么會調用名字帶有”stret”的函數(shù)。排列組合正好四個方法。

? ? ? ? 值得一提的是在 i386 平臺處理返回類型為浮點數(shù)的消息時,需要用到objc_msgSend_fpret函數(shù)來進行處理,這是因為返回類型為浮點數(shù)的函數(shù)對應的 ABI(Application Binary Interface) 與返回整型的函數(shù)的 ABI 不兼容。此時objc_msgSend不再適用,于是objc_msgSend_fpret被派上用場,它會對浮點數(shù)寄存器做特殊處理。不過在 PPC 或 PPC64 平臺是不需要麻煩它的。?

? ? ? ? PS:有木有發(fā)現(xiàn)這些函數(shù)的命名規(guī)律哦?帶“Super”的是消息傳遞給超類;“stret”可分為“st”+“ret”兩部分,分別代表“struct”和“return”;“fpret”就是“fp”+“ret”,分別代表“floating-point”和“return”。

?

? ? ? ? ?下面詳細敘述下消息發(fā)送步驟:

  • 檢測這個 selector 是不是要忽略的。比如 Mac OS X 開發(fā),有了垃圾回收就不理會 retain, release 這些函數(shù)了。
  • 檢測這個 target 是不是 nil 對象。ObjC 的特性是允許對一個 nil 對象執(zhí)行任何一個方法不會 Crash,因為會被忽略掉。
  • 如果上面兩個都過了,那就開始查找這個類的 IMP,先從 cache 里面找,完了找得到就跳到對應的函數(shù)去執(zhí)行。
  • 如果 cache 找不到就找一下方法分發(fā)表。
  • 如果分發(fā)表找不到就到超類的分發(fā)表去找,一直找,直到找到NSObject類為止。
  • 如果還找不到就要開始進入動態(tài)方法解析了。

PS:這里說的分發(fā)表其實就是Class中的方法列表,它將方法選擇器和方法實現(xiàn)地址聯(lián)系起來。

? ?4.2.方法中的隱藏參數(shù)

   ??我們經常在方法中使用self關鍵字來引用實例本身,但從沒有想過為什么self就能取到調用當前對象的方法吧。其實self的內容是在方法運行時被偷偷的動態(tài)傳入的。(之所以說它們是隱藏的是因為在源代碼方法的定義中并沒有聲明這兩個參數(shù)。它們是在代碼被編譯時被插入實現(xiàn)中的。盡管這些參數(shù)沒有被明確聲明,在源代碼中我們仍然可以引用它們。)

?

? ? ? ? ?當objc_msgSend找到方法對應的實現(xiàn)時,它將直接調用該方法實現(xiàn),并將消息中所有的參數(shù)都傳遞給方法實現(xiàn),同時,它還將傳遞兩個隱藏的參數(shù):

?

  • 接收消息的對象(也就是self指向的內容)
  • 方法選擇器(_cmd指向的內容)

    而當方法中的super關鍵字接收到消息時,編譯器會創(chuàng)建一個objc_super結構體:

    struct objc_super { id receiver; Class class; };

    ?

    這個結構體指明了消息應該被傳遞給特定超類的定義。但receiver仍然是self本身,這點需要注意,因為當我們想通過[super class]獲取超類時,編譯器只是將指向selfid指針和classSEL傳遞給了objc_msgSendSuper函數(shù),因為只有在NSObject類才能找到class方法,然后class方法調用object_getClass(),接著調用objc_msgSend(objc_super->receiver, @selector(class)),傳入的第一個參數(shù)是指向selfid指針,與調用[self class]相同,所以我們得到的永遠都是self的類型。

?

? ?4.3.獲取方法地址

   ? ??在IMP那節(jié)提到過可以避開消息綁定而直接獲取方法的地址并調用方法。這種做法很少用,除非是需要持續(xù)大量重復調用某方法的極端情況,避開消息發(fā)送泛濫而直接調用該方法會更高效。

?

5.動態(tài)方法解析

? ? ? ?可以動態(tài)地提供一個方法的實現(xiàn)。例如我們可以用@dynamic關鍵字在類的實現(xiàn)文件中修飾一個屬性:

  @dynamic propertyName;

(@dynamic 意思是由開發(fā)人員提供相應的代碼:對于只讀屬性需要提供 getter,對于讀寫屬性需要提供 getter 和setter。

@synthesize 意思是,除非開發(fā)人員已經做了,否則由編譯器生成相應的代碼,以滿足屬性聲明。)

? ? ?可以通過分別重載resolveInstanceMethod:resolveClassMethod:方法分別添加實例方法實現(xiàn)和類方法實現(xiàn)。因為當 Runtime 系統(tǒng)在Cache和方法分發(fā)表中(包括超類)找不到要執(zhí)行的方法時,Runtime會調用resolveInstanceMethod:resolveClassMethod:來給程序員一次動態(tài)添加方法實現(xiàn)的機會。

//RuntimeMain.h文件

?

// // RuntimeMain.h // RuntimeTest // // Created by Vie on 2017/2/8. // Copyright ? 2017年 Vie. All rights reserved. // #import <Foundation/Foundation.h>@interface RuntimeMain : NSObject@end

?

?

?

?

//RuntimeMain.m文件

?

// // RuntimeMain.m // RuntimeTest // // Created by Vie on 2017/2/8. // Copyright ? 2017年 Vie. All rights reserved. // #import "RuntimeMain.h" #import <objc/runtime.h> #import "RuntimeFoward.h" @implementation RuntimeMain #pragma mark 實例方法動態(tài)解析重定向 //動態(tài)實例方法解析,如果這里沒有找到該執(zhí)行的方法會指定到重定向forwardingTargetForSelector:方法 +(BOOL)resolveInstanceMethod:(SEL)sel{if (sel == @selector(goToSchool:)) {//用class_addMethod函數(shù)完成向特定類添加特定方法實現(xiàn)的操作//其中 “v@:” 表示返回值和參數(shù)(為了兼容32位機型使用"v@:@"),增加f處理float參數(shù),這個符號涉及 Type Encodingreturn class_addMethod([self class], sel, class_getMethodImplementation([self class], @selector(myInstanceMethod:)), "v@:@");}return [super resolveInstanceMethod:sel]; } //重定向實例方法,Lenaer實現(xiàn)了該實例方法;如果forwardingTargetForSelector:未找方法就轉發(fā)給forwardInvocation:方法 //forwardingTargetForSelector:僅支持一個對象的返回,也就是說消息只能被轉發(fā)給一個對象 -(id)forwardingTargetForSelector:(SEL)aSelector{if(aSelector == @selector(learnClass:)) {return [[NSClassFromString(@"RuntimeFoward") alloc] init];}//千萬別返回self,因為那樣會死循環(huán)。重定向的類未實現(xiàn)該方法會導致崩潰return [super forwardingTargetForSelector:aSelector]; } //在forwardInvocation:之前創(chuàng)建一個有效的方法簽名 -(NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{//其中 “v@:” 表示返回值和參數(shù)(為了兼容32位機型使用"v@:@"),增加f處理float參數(shù),這個符號涉及 Type Encodingreturn [NSMethodSignature signatureWithObjCTypes:"v@:@"]; }//如果resolveInstanceMethod:和forwardingTargetForSelector:以及forwardInvocation:都未找到方法實現(xiàn)將崩潰,所以可以再最后else定義一個錯誤日志輸出方法處理崩潰//forwardInvocation:可以將消息同時轉發(fā)給任意多個對象//forwardInvocation:方法就像一個不能識別的消息的分發(fā)中心,將這些消息轉發(fā)給不同接收對象。或者它也可以象一個運輸站將所有的消息都發(fā)送給同一個接收對象。它可以將一個消息翻譯成另外一個消息,或者簡單的”吃掉“某些消息,因此沒有響應也沒有錯誤。forwardInvocation:方法也可以對不同的消息提供同樣的響應,這一切都取決于方法的具體實現(xiàn)。該方法所提供是將不同的對象鏈接到消息鏈的能力。 -(void)forwardInvocation:(NSInvocation *)anInvocation{return [anInvocation invokeWithTarget:[[RuntimeFoward alloc] init]]; }-(void)myInstanceMethod:(NSString *)string{NSLog(@"myInstanceMethod = %@", string);}#pragma mark 類方法動態(tài)解析重定向//動態(tài)類方法解析,如果這里沒有找到該執(zhí)行的方法會指定到重定向forwardingTargetForSelector:方法 +(BOOL)resolveClassMethod:(SEL)sel{if (sel==@selector(classFouction:)) {//用class_addMethod函數(shù)完成向特定類添加特定方法實現(xiàn)的操作//其中 “v@:” 表示返回值和參數(shù)(為了兼容32位機型使用"v@:@")增加f處理float參數(shù),這個符號涉及 Type Encodingreturn class_addMethod(object_getClass(self), sel, class_getMethodImplementation(object_getClass(self), @selector(myClassMethod:)), "v@:@f");}return [class_getSuperclass(self) resolveClassMethod:sel];}//重定向類方法,Lenaer實現(xiàn)了該類方法;如果resolveClassMethod:和forwardingTargetForSelector:都未找到方法實現(xiàn)將崩潰,所以可以再最后else定義一個錯誤日志輸出方法處理崩潰 +(id)forwardingTargetForSelector:(SEL)aSelector{if(aSelector == @selector(testNotMe:)) {return NSClassFromString(@"RuntimeFoward") ;}//千萬別返回self,因為那樣會死循環(huán)。重定向的類未實現(xiàn)該方法會導致崩潰return [super forwardingTargetForSelector:aSelector]; } +(void)myClassMethod:(NSString *)string{NSLog(@"myClassMethod = %@", string);} @end

?

?

?

//RuntimeFoward.h文件

// // RuntimeFoward.h // RuntimeTest // // Created by Vie on 2017/2/8. // Copyright ? 2017年 Vie. All rights reserved. // #import <Foundation/Foundation.h>@interface RuntimeFoward : NSObject @end

?

//RuntimeFoward.m文件

// // RuntimeFoward.m // RuntimeTest // // Created by Vie on 2017/2/8. // Copyright ? 2017年 Vie. All rights reserved. // #import "RuntimeFoward.h"@implementation RuntimeFoward -(void)fowardGet{NSLog(@"消息轉發(fā)給RuntimeFoward"); }-(void)fowardWithString:(NSString *)string{NSLog(@"消息轉發(fā)給RuntimeFoward,并帶參數(shù)%@",string); } -(void)learnClass:(NSString *)string{NSLog(@"消息轉發(fā)給RuntimeFoward,learnClass并帶參數(shù)%@",string); }-(NSString *)getInfo:(NSString *)name height:(float)aHeight{return [NSString stringWithFormat:@"%@身高%f",name,aHeight]; } -(float)getRectangularArea:(float)aWidth height:(float)aHeight{return aWidth*aHeight; }+(void)testNotMe:(NSString *)string{NSLog(@"消息轉發(fā)給RuntimeFoward,testNotMe并帶參數(shù)%@",string); }@end

?

?使用

#import <objc/message.h> #import "RuntimeMain.h"//調用實例方法RuntimeMain *runMain=[[RuntimeMain alloc] init];//調用無參數(shù)無返回值方法((void (*) (id, SEL)) objc_msgSend) (runMain, sel_registerName("fowardGet"));//調用有參數(shù)無返回值方法((void (*) (id, SEL,NSString *)) objc_msgSend)(runMain,sel_registerName("fowardWithString:"),@"哈哈哈");((void (*) (id, SEL,NSString *)) objc_msgSend)(runMain,sel_registerName("goToSchool:"),@"哈哈哈");((void (*) (id, SEL,NSString *)) objc_msgSend)(runMain,sel_registerName("learnClass:"),@"哈哈哈");//調用返回String有參數(shù)方法NSString *infoString= ((NSString* (*)(id,SEL,NSString *,float)) objc_msgSend)(runMain,sel_registerName("getInfo:height:"),@"張三",175.81);NSLog(@"%@",infoString);//調用返回float方法, float area=((float (*) (id,SEL,float,float)) objc_msgSend)(runMain,sel_registerName("getRectangularArea:height:"),12.0f,12.0f); NSLog(@"獲得長方形面積為%f",area);//調用類方法[RuntimeMain performSelector:@selector(classFouction:) withObject:@"xxx"];[RuntimeMain performSelector:@selector(testNotMe:) withObject:@"xxx"];

運行結果:

2017-02-08 19:33:30.608 RuntimeTest[26467:721253] 消息轉發(fā)給RuntimeFoward 2017-02-08 19:33:30.609 RuntimeTest[26467:721253] 消息轉發(fā)給RuntimeFoward,并帶參數(shù)哈哈哈 2017-02-08 19:33:30.609 RuntimeTest[26467:721253] myInstanceMethod = 哈哈哈 2017-02-08 19:33:30.610 RuntimeTest[26467:721253] 消息轉發(fā)給RuntimeFoward,learnClass并帶參數(shù)哈哈哈 2017-02-08 19:33:30.611 RuntimeTest[26467:721253] 張三身高175.809998 2017-02-08 19:33:30.612 RuntimeTest[26467:721253] 獲得長方形面積為144.000000 2017-02-08 19:33:30.613 RuntimeTest[26467:721253] myClassMethod = xxx 2017-02-08 19:33:30.616 RuntimeTest[26467:721253] 消息轉發(fā)給RuntimeFoward,testNotMe并帶參數(shù)xxx

?

?

6.消息轉發(fā)

? ?6.1.重定向

   在消息轉發(fā)機制執(zhí)行前,Runtime 系統(tǒng)會再給我們一次偷梁換柱的機會

替換對象方法接受者- (id)forwardingTargetForSelector:(SEL)aSelector方法替換消息的接受者為其他對象:

替換類方法的接受者,需要覆寫?+ (id)forwardingTargetForSelector:(SEL)aSelector?方法,并返回類對象

?

? ?6.2.轉發(fā)

?

當動態(tài)方法解析未找到執(zhí)行方法,消息轉發(fā)機制會被觸發(fā)。先methodSignatureForSelector:創(chuàng)建有效簽名,再forwardInvocation:執(zhí)行方法,我們可以重寫這個方法來定義我們的轉發(fā)邏輯:

?

?

?

? ?6.3.轉發(fā)和多繼承

? ? ? ? ? 轉發(fā)和繼承相似,可以用于為Objc編程添加一些多繼承的效果。就像下圖那樣,一個對象把消息轉發(fā)出去,就好似它把另一個對象中的方法借過來或是“繼承”過來一樣。

這使得不同繼承體系分支下的兩個類可以“繼承”對方的方法,在上圖中Warrior和Diplomat沒有繼承關系,但是Warrior將negotiate消息轉發(fā)給了Diplomat后,就好似Diplomat是Warrior的超類一樣。?

? ? ? ? ?消息轉發(fā)彌補了 Objc 不支持多繼承的性質,也避免了因為多繼承導致單個類變得臃腫復雜。它將問題分解得很細,只針對想要借鑒的方法才轉發(fā),而且轉發(fā)機制是透明的。

? ?6.4.替代者對象(Surrogate Objects)

? ? ? ? ? 轉發(fā)不僅能模擬多繼承,也能使輕量級對象代表重量級對象。弱小的女人背后是強大的男人,畢竟女人遇到難題都把它們轉發(fā)給男人來做了。這里有一些適用案例,可以參看官方文檔。

? ?6.5.轉發(fā)與繼承

? ? ? ? ? 盡管轉發(fā)很像繼承,但是NSObject類不會將兩者混淆。像respondsToSelector:?和?isKindOfClass:這類方法只會考慮繼承體系,不會考慮轉發(fā)鏈。?

7.健壯的實例變量(Non Fragile ivars)? ?

? ? ? ? ?在 Runtime 的現(xiàn)行版本中,最大的特點就是健壯的實例變量。當一個類被編譯時,實例變量的布局也就形成了,它表明訪問類的實例變量的位置。從對象頭部開始,實例變量依次根據(jù)自己所占空間而產生位移:

上圖左邊是NSObject類的實例變量布局,右邊是我們寫的類的布局,也就是在超類后面加上我們自己類的實例變量,看起來不錯。但試想如果哪天蘋果更新了NSObject類,發(fā)布新版本的系統(tǒng)的話,那就悲劇了:??

我們自定義的類被劃了兩道線,那是因為那塊區(qū)域跟超類重疊了。唯有蘋果將超類改為以前的布局才能拯救我們,但這樣也導致它們不能再拓展它們的框架了,因為成員變量布局被死死地固定了。在脆弱的實例變量(Fragile ivars) 環(huán)境下我們需要重新編譯繼承自 Apple 的類來恢復兼容性。那么在健壯的實例變量下會發(fā)生什么呢?

在健壯的實例變量下編譯器生成的實例變量布局跟以前一樣,但是當 runtime 系統(tǒng)檢測到與超類有部分重疊時它會調整你新添加的實例變量的位移,那樣你在子類中新添加的成員就被保護起來了。?

需要注意的是在健壯的實例變量下,不要使用sizeof(SomeClass),而是用class_getInstanceSize([SomeClass class])代替;也不要使用offsetof(SomeClass, SomeIvar),而要用ivar_getOffset(class_getInstanceVariable([SomeClass class], "SomeIvar"))來代替。

8.Objective-C Associated Objects

? ? ?在 OS X 10.6 之后,Runtime系統(tǒng)讓Objc支持向對象動態(tài)添加變量。涉及到的函數(shù)有以下三個:

void objc_setAssociatedObject ( id object, const void *key, id value, objc_AssociationPolicy policy ); id objc_getAssociatedObject ( id object, const void *key ); void objc_removeAssociatedObjects ( id object );

//這些方法以鍵值對的形式動態(tài)地向對象添加、獲取或刪除關聯(lián)值。其中關聯(lián)政策是一組枚舉常量

enum {OBJC_ASSOCIATION_ASSIGN = 0,OBJC_ASSOCIATION_RETAIN_NONATOMIC = 1,OBJC_ASSOCIATION_COPY_NONATOMIC = 3,OBJC_ASSOCIATION_RETAIN = 01401,OBJC_ASSOCIATION_COPY = 01403 };

這些常量對應著引用關聯(lián)值的政策,也就是 Objc 內存管理的引用計數(shù)機制。

//例

//h文件

#import "TKIMRoom.h"@interface TKIMRoom(TKIMRoomExt) @property(nonatomic,copy)NSString *islock; @end

?

//m文件

#import "TKIMRoomExt.h" #import <objc/runtime.h> @implementation TKIMRoom(TKIMRoomExt) -(NSString *)islock{return objc_getAssociatedObject(self, "islock"); } -(void)setIslock:(NSString *)islock{objc_setAssociatedObject(self, "islock", islock, OBJC_ASSOCIATION_RETAIN_NONATOMIC); } @end

?

9.Method Swizzling

? ? ? ?之前所說的消息轉發(fā)雖然功能強大,但需要我們了解并且能更改對應類的源代碼,因為我們需要實現(xiàn)自己的轉發(fā)邏輯。當我們無法觸碰到某個類的源代碼,卻想更改這個類某個方法的實現(xiàn)時,該怎么辦呢?可能繼承類并重寫方法是一種想法,但是有時無法達到目的。這里介紹的是 Method Swizzling 用于改變一個已經存在的 selector 的實現(xiàn)。這項技術使得在運行時通過改變 selector 在類的消息分發(fā)列表中的映射從而改變方法的掉用成為可能。跟消息轉發(fā)相比,Method Swizzling 的做法更為隱蔽,甚至有些冒險,也增大了debug的難度。

?

//例1:

//NSArray+Swizzle.h文件

// // NSArray+Swizzle.m // RuntimeTest // 通過分類NSarry達到在調用時候,替換方法實現(xiàn) // Created by Vie on 2017/2/9. // Copyright ? 2017年 Vie. All rights reserved. // #import "NSArray+Swizzle.h"@implementation NSArray (Swizzle) -(id)myLastObject{//別忘記這是我們準備調換IMP的selector,[self myLastObject] 將會執(zhí)行真的 [self lastObject] 。id result=[self myLastObject];NSLog(@"**********替換實現(xiàn)了myLastObject**********");return result; } @end

//NSArray+Swizzle.m文件

// // NSArray+Swizzle.m // RuntimeTest // 通過分類NSarry達到在調用時候,替換方法實現(xiàn) // Created by Vie on 2017/2/9. // Copyright ? 2017年 Vie. All rights reserved. // #import "NSArray+Swizzle.h"@implementation NSArray (Swizzle) -(id)myLastObject{//別忘記這是我們準備調換IMP的selector,[self myLastObject] 將會執(zhí)行真的 [self lastObject] 。id result=[self myLastObject];NSLog(@"**********替換實現(xiàn)了myLastObject**********");return result; } @end

//在main.m文件調用

// // main.m // RuntimeTest // // Created by Vie on 2017/2/7. // Copyright ? 2017年 Vie. All rights reserved. // #import <UIKit/UIKit.h> #import "AppDelegate.h"#import <objc/message.h> #import "NSArray+Swizzle.h"int main(int argc, char * argv[]) {@autoreleasepool { #pragma mark Method Swizzling測試Method original_Method=class_getInstanceMethod([NSArray class], @selector(lastObject));Method swizzling_Method=class_getInstanceMethod([NSArray class], @selector(myLastObject));method_exchangeImplementations(original_Method, swizzling_Method);NSArray *array = @[@"0",@"1",@"2",@"3"];//準備調換IMP的selector,[self lastObject] 將會執(zhí)行真的 [self myLastObject]。NSString *string = [array lastObject];NSLog(@"TEST RESULT : %@",string);return 0;} }

//調用結果

2017-02-09 11:06:02.615 RuntimeTest[39348:924342] **********替換實現(xiàn)了myLastObject********** 2017-02-09 11:06:02.616 RuntimeTest[39348:924342] TEST RESULT : 3

?

//例2

//UIViewController+Swizzling.h文件

//給UIViewController的viewDidLoad方法進行category

// // UIViewController+Swizzling.h // RuntimeTest // 分類UIViewController,在的+(void)load方法中添加Method Swizzling方法,可以實現(xiàn)頁面統(tǒng)計的需求 // Created by Vie on 2017/2/9. // Copyright ? 2017年 Vie. All rights reserved. // #import <UIKit/UIKit.h>@interface UIViewController (Swizzling)@end

?

//UIViewController+Swizzling.m文件

// // UIViewController+Swizzling.m // RuntimeTest // 分類UIViewController,在的+(void)load方法中添加Method Swizzling方法,可以實現(xiàn)頁面統(tǒng)計的需求 // Created by Vie on 2017/2/9. // Copyright ? 2017年 Vie. All rights reserved. // #import "UIViewController+Swizzling.h" #import <objc/runtime.h> @implementation UIViewController (Swizzling) //由于load類方法是程序運行時這個類被加載到內存中就調用的一個方法,執(zhí)行比較早,并且不需要我們手動調用。而且這個方法具有唯一性,也就是只會被調用一次,不用擔心資源搶奪的問題。 +(void)load{[super load];//通過class_getInstanceMethod()函數(shù)從當前對象中的method list獲取method結構體;//如果是類方法就使用class_getClassMethod()函數(shù)獲取。Method original_Method=class_getInstanceMethod([self class], @selector(viewDidLoad));Method swizzling_Method=class_getInstanceMethod([self class], @selector(swizzling_viewDidLoad));/*** 我們在這里使用class_addMethod()函數(shù)對Method Swizzling做了一層驗證,如果self沒有實現(xiàn)被交換的方法,會導致失敗。* 而且self沒有交換的方法實現(xiàn),但是父類有這個方法,這樣就會調用父類的方法,結果就不是我們想要的結果了。* 所以我們在這里通過class_addMethod()的驗證,如果self實現(xiàn)了這個方法,class_addMethod()函數(shù)將會返回NO,我們就可以對其進行交換了。*/if (!class_addMethod([self class], @selector(viewDidLoad), method_getImplementation(swizzling_Method), method_getTypeEncoding(swizzling_Method))) {method_exchangeImplementations(original_Method, swizzling_Method);} } -(void)swizzling_viewDidLoad{NSString *str=[NSString stringWithFormat:@"%@",self.class];//將系統(tǒng)的UIViewController對象剔除掉if (![str containsString:@"UI"]) {NSLog(@"\n*************統(tǒng)計打點進入頁面:%@",str);}//系統(tǒng)調用UIViewController的viewDidLoad方法時,實際上執(zhí)行的是我們實現(xiàn)的swizzling_viewDidLoad方法。而我們在調用[self swizzling_viewDidLoad];時,執(zhí)行的是UIViewController的viewDidLoad方法。 [self swizzling_viewDidLoad]; } @end

//在試圖控制器里面導入頭文件就可使用

#import "UIViewController+Swizzling.h"

//調用結果

2017-02-09 12:58:52.411 RuntimeTest[40579:967313] *************統(tǒng)計打點進入頁面:ViewController

? ? ? ?9.1.Method Swizzling類簇

?

? ? ? ? ? ? ?在我們項目開發(fā)過程中,經常因為NSArray數(shù)組越界或者NSDictionary的key或者value值為nil等問題導致的崩潰,對于這些問題蘋果并不會報一個警告,而是直接崩潰。由此,我們可以根據(jù)上面所學,對NSArray、NSMutableArray、NSDictionary、NSMutableDictionary等類進行Method Swizzling,實現(xiàn)方式還是按照上面的例子來做。但是....你發(fā)現(xiàn)Method Swizzling根本就不起作用,代碼也沒寫錯啊,到底是什么鬼?

?

? ? ? ? ? ? ?這是因為Method Swizzling對NSArray這些的類簇是不起作用的。因為這些類簇類,其實是一種抽象工廠的設計模式。抽象工廠內部有很多其它繼承自當前類的子類,抽象工廠類會根據(jù)不同情況,創(chuàng)建不同的抽象對象來進行使用。例如我們調用NSArray的objectAtIndex:方法,這個類會在方法內部判斷,內部創(chuàng)建不同抽象類進行操作。所以也就是我們對NSArray類進行操作其實只是對父類進行了操作,在NSArray內部會創(chuàng)建其他子類來執(zhí)行操作,真正執(zhí)行操作的并不是NSArray自身,所以我們應該對其“真身”進行操作。

//例

//NSArray+Swizzling.h文件

// // NSArray+Swizzling.h // RuntimeTest // // Created by Vie on 2017/2/9. // Copyright ? 2017年 Vie. All rights reserved. // #import <Foundation/Foundation.h>@interface NSArray (Swizzling)@end

?

//NSArray+Swizzling.m文件

// // NSArray+Swizzling.m // RuntimeTest // // Created by Vie on 2017/2/9. // Copyright ? 2017年 Vie. All rights reserved. // #import "NSArray+Swizzling.h" #import <objc/runtime.h> @implementation NSArray (Swizzling) +(void)load{[super load];//通過class_getInstanceMethod()函數(shù)從當前對象中的method list獲取method結構體;//如果是類方法就使用class_getClassMethod()函數(shù)獲取。Method original_Method=class_getInstanceMethod(objc_getClass("__NSArrayI"), @selector(objectAtIndex:));Method swizzling_Method=class_getInstanceMethod(objc_getClass("__NSArrayI"), @selector(swizzling_objectAtIndex:));/*** 我們在這里使用class_addMethod()函數(shù)對Method Swizzling做了一層驗證,如果self沒有實現(xiàn)被交換的方法,會導致失敗。* 而且self沒有交換的方法實現(xiàn),但是父類有這個方法,這樣就會調用父類的方法,結果就不是我們想要的結果了。* 所以我們在這里通過class_addMethod()的驗證,如果self實現(xiàn)了這個方法,class_addMethod()函數(shù)將會返回NO,我們就可以對其進行交換了。*/if (!class_addMethod([self class], @selector(objectAtIndex:), method_getImplementation(swizzling_Method), method_getTypeEncoding(swizzling_Method))) {method_exchangeImplementations(original_Method, swizzling_Method);}} - (id)swizzling_objectAtIndex:(NSUInteger)index {NSUInteger count=self.count;if (count-1 < index) {//數(shù)組下標越界后執(zhí)行NSLog(@"\n*************%s數(shù)組下標越界 %s", class_getName(self.class), __func__);return nil;} else {//數(shù)組下標沒越界就調用swizzling_objectAtIndex:執(zhí)行objectAtIndex:return [self swizzling_objectAtIndex:index];} } @end

?

//在main.m文件調用

// // main.m // RuntimeTest // // Created by Vie on 2017/2/7. // Copyright ? 2017年 Vie. All rights reserved. // #import <UIKit/UIKit.h> #import "AppDelegate.h"int main(int argc, char * argv[]) {@autoreleasepool { #pragma mark Method Swizzling測試2NSArray *array = @[@"0",@"1",@"2",@"3"];[array objectAtIndex:4];return 0;} }

?

//運行結果

2017-02-09 13:29:56.050 RuntimeTest[41085:985169] *************__NSArrayI數(shù)組下標越界 -[NSArray(Swizzling) swizzling_objectAtIndex:]

?

? ? ? ?9.2.Method Swizzling 錯誤剖析

? ? ? 在上面的例子中,如果只是單獨對NSArray或NSMutableArray中的單個類進行Method Swizzling,是可以正常使用并且不會發(fā)生異常的。如果進行Method Swizzling的類中,有兩個類有繼承關系的,并且Swizzling了同一個方法。例如同時對NSArray和NSMutableArray中的objectAtIndex:方法都進行了Swizzling,這樣可能會導致父類Swizzling失效的問題。

? ? ? 對于這種問題主要是兩個原因導致的,首先是不要在+ (void)load方法中調用[super load]方法,這會導致父類的Swizzling被重復執(zhí)行兩次,這樣父類的Swizzling就會失效。例如下面的兩張圖片,你會發(fā)現(xiàn)由于NSMutableArray調用了[super load]導致父類NSArray的Swizzling代碼被執(zhí)行了兩次。

? ? ?還有一個原因就是因為代碼邏輯導致Swizzling代碼被執(zhí)行了多次,這也會導致Swizzling失效,其實原理和上面的問題是一樣的。

10.總結

? ? ? ? ? 我們之所以讓自己的類繼承NSObject不僅僅因為蘋果幫我們完成了復雜的內存分配問題,更是因為這使得我們能夠用上 Runtime 系統(tǒng)帶來的便利。可能我們平時寫代碼時可能很少會考慮一句簡單的[receiver message]背后發(fā)生了什么,而只是當做方法或函數(shù)調用。深入理解 Runtime 系統(tǒng)的細節(jié)更有利于我們利用消息機制寫出功能更強大的代碼,比如 Method Swizzling 等。

轉載于:https://www.cnblogs.com/douniwanxia/p/6374157.html

總結

以上是生活随笔為你收集整理的iOS,Objective-C Runtime的全部內容,希望文章能夠幫你解決所遇到的問題。

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

中文字幕精品一区二区三区电影 | 91污视频在线观看 | 欧美日韩视频网站 | 欧美日韩亚洲在线观看 | 日韩欧美在线综合网 | av丁香花| 国产精品99久久久精品 | 大胆欧美gogo免费视频一二区 | 成人国产电影在线观看 | 狠狠干成人 | 91成人观看 | 欧美亚洲成人xxx | 亚洲一区视频免费观看 | 亚洲综合射 | free. 性欧美.com| 在线直播av | 欧美激情视频一区二区三区免费 | 深夜免费网站 | 热久久电影 | 国产成人精品综合久久久久99 | 精品一区在线看 | 国产成人免费观看久久久 | 免费观看福利视频 | 国内精品久久久久影院一蜜桃 | 国语久久| 久久久精品欧美一区二区免费 | 天天操天天摸天天爽 | 国产亚洲精品bv在线观看 | 人人爽人人澡 | 天天狠狠| 夜夜骑首页 | 青青五月天 | 2019精品手机国产品在线 | 综合激情 | 男女激情片在线观看 | 久久草网站 | 91网在线观看 | 久久黄色免费 | 六月丁香激情综合色啪小说 | 密桃av在线 | 天天操天天射天天爽 | 成人a视频片观看免费 | 久草在线手机视频 | 在线看v片成人 | 永久av免费在线观看 | 国产99久久精品 | 亚洲精品永久免费视频 | 区一区二区三在线观看 | 日本中文字幕在线看 | 日韩xxxxxxxxx| 午夜精品久久一牛影视 | 欧美日韩国产色综合一二三四 | 波多野结衣一区三区 | 国产麻豆电影 | 欧美一区日韩精品 | 日韩理论在线观看 | 日韩在线视频在线观看 | 午夜在线日韩 | 狠狠成人 | www.香蕉| 免费麻豆网站 | 色wwwww| 国产成人不卡 | 91麻豆网站 | 一个色综合网站 | 色在线观看网站 | 91在线在线观看 | 国产区在线 | 岛国精品一区二区 | 国产亚洲欧美精品久久久久久 | 久久久免费精品视频 | 黄色91免费观看 | 一区二区理论片 | 久久精品牌麻豆国产大山 | 91片黄在线观 | 日韩精品1区2区 | 国产一区二区中文字幕 | 欧美日韩国产免费视频 | 天天操欧美| 亚洲码国产日韩欧美高潮在线播放 | 中文字幕乱码视频 | 亚洲精品在线一区二区三区 | 亚洲视频 在线观看 | 亚洲精品在线视频播放 | 日韩av一区二区三区 | 午夜精品一区二区三区四区 | 日韩欧美高清一区二区三区 | 在线观看va| 免费观看的黄色片 | 激情视频国产 | 免费麻豆 | 欧美成天堂网地址 | 中文字幕在线观看视频一区二区三区 | 国产伦理精品一区二区 | 久草网免费 | 亚洲精品视频在线播放 | 久久综合九色综合久久久精品综合 | 欧美精品首页 | 国产在线观看网站 | 天堂av在线中文在线 | 98精品国产自产在线观看 | 在线观看黄网 | 在线观看不卡的av | 91在线观看视频网站 | 天天射射天天 | 最近乱久中文字幕 | 天天操天天干天天操天天干 | 国产日产精品久久久久快鸭 | 97超级碰碰 | 五月天综合网站 | 国产精品成 | 在线免费三级 | 91成人看片 | 黄色成人av在线 | 国产精品毛片一区二区在线看 | 久久久久免费视频 | 亚洲精品综合一二三区在线观看 | 色妞色视频一区二区三区四区 | 午夜久久福利影院 | 国产专区视频在线 | 久久久精品国产免费观看一区二区 | 色播五月婷婷 | 欧美性爽爽 | 玖玖色在线观看 | 999精品网| 一区二区三区精品在线 | 午夜 久久 tv | 91视频 - v11av | 日本高清中文字幕有码在线 | 欧美视频www | 日韩黄色免费 | 午夜视频导航 | 高清久久久 | 日韩电影在线观看一区二区三区 | 麻豆视频在线免费 | 亚洲午夜久久久久久久久电影网 | aⅴ视频在线 | 久草免费在线 | 日韩免费高清在线 | 午夜视频在线观看一区二区三区 | 日韩精品一区二区三区不卡 | 国产亚洲人成网站在线观看 | av网站免费看 | 在线播放精品一区二区三区 | 91你懂的| 久久蜜臀一区二区三区av | 成人播放器 | 97国产精品| 性色xxxxhd| 黄色成品视频 | 五月激情av| 99热超碰| 一本一本久久a久久精品综合 | 亚洲精品网址在线观看 | 狠狠操狠狠 | 十八岁免进欧美 | 欧美日韩中文字幕综合视频 | 午夜精品一区二区三区在线视频 | 成年人三级网站 | 精品福利国产 | 日韩精品一区二区免费视频 | 日韩精品在线视频免费观看 | 911久久香蕉国产线看观看 | 精品视频999| 九九九视频在线 | 亚洲va欧美 | 在线观看免费高清视频大全追剧 | 亚洲激情电影在线 | 亚洲精品国产欧美在线观看 | 国产视频丨精品|在线观看 国产精品久久久久久久久久久久午夜 | 2019中文最近的2019中文在线 | 国产精品伦一区二区三区视频 | 天天操天天摸天天干 | 亚洲成人动漫在线观看 | 欧美在线1 | 精品国产乱码久久 | 中文字幕丝袜 | 波多野结衣最新 | 成人黄色中文字幕 | 午夜成人影视 | 国产精品午夜久久久久久99热 | 综合伊人av | 日日操日日 | 国产剧情一区二区 | 亚洲热久久| 免费电影一区二区三区 | 美女视频久久久 | 欧美亚洲成人免费 | 亚洲 欧美 精品 | 久久99在线视频 | 日韩在线观看第一页 | 久久激情五月婷婷 | av在线亚洲天堂 | 亚洲欧洲日韩在线观看 | 伊人激情网 | 99久久婷婷国产综合精品 | 午夜视频在线观看网站 | 久久歪歪 | 欧美最新大片在线看 | 激情视频二区 | 色综合久久88色综合天天6 | 亚洲视频免费在线看 | 中文字幕制服丝袜av久久 | 中文字幕色综合网 | 欧美久久久一区二区三区 | 激情小说 五月 | 欧美精品亚洲精品日韩精品 | 深爱激情久久 | 性色视频在线 | 亚洲精品videossex少妇 | 激情影音先锋 | 一级免费黄色 | 99久久精品久久久久久清纯 | 高清国产午夜精品久久久久久 | 国产三级视频在线 | 亚洲精品国产精品国自产观看浪潮 | 一级黄色片网站 | 久久久av免费| 中文字幕免费不卡视频 | 国色天香在线 | 国产成人高清在线 | 婷婷国产在线观看 | 国产精品综合在线观看 | 成 人 免费 黄 色 视频 | 日韩丝袜 | 国产亚洲人 | 91视频 - 114av| 色老板在线视频 | 五月激情在线 | 亚洲人人av| 午夜婷婷综合 | 国产亚洲视频中文字幕视频 | 久久综合久久综合这里只有精品 | 欧美a级在线免费观看 | 97成人精品视频在线播放 | 久草综合视频 | 91视频久久久久 | 91成人精品一区在线播放 | 黄色网在线免费观看 | 国际精品网 | 色黄久久久久久 | 婷婷去俺也去六月色 | 91av在线播放 | 伊人av综合 | 精品中文字幕视频 | 成人一区二区三区在线观看 | 国产精品自拍av | 免费福利小视频 | 国产亚洲一区 | 日三级在线| 中文字幕91| 中文字幕在线资源 | 青草视频在线 | 久久久久久黄 | 国产精品国内免费一区二区三区 | 久久精品激情 | 国产精品国产三级国产专区53 | 91麻豆国产 | 久久婷五月 | 日韩免费区 | 国产99久久久精品 | 超碰97人人在线 | 婷婷丁香色综合狠狠色 | 国产视频在线观看免费 | 一区二区三区手机在线观看 | 欧美日韩在线观看一区二区 | 日日夜夜精品视频天天综合网 | 国产精品国产自产拍高清av | 免费一级片观看 | 久久66热这里只有精品 | 美女免费av | 色多多视频在线观看 | 色婷婷福利 | 99久高清在线观看视频99精品热在线观看视频 | 日韩av在线免费看 | 久久这里只有精品视频99 | 在线免费色视频 | 黄色视屏av | 亚洲欧美乱综合图片区小说区 | 亚洲成av人片在线观看无 | 精品91视频 | 国产亚洲精品女人久久久久久 | 婷婷色伊人| 久久国产一区二区 | 综合色亚洲 | 国产精品18久久久久久不卡孕妇 | 国产精品九九久久久久久久 | 国产精品免费大片视频 | 中文字幕一区二区三区视频 | 亚洲视频免费在线观看 | 国产日本亚洲高清 | 99中文字幕在线观看 | 黄色在线观看免费 | 久久久免费毛片 | 最新av网址在线 | 久久国产精品久久久久 | 国产小视频免费在线观看 | bayu135国产精品视频 | a级片在线播放 | 一级精品视频在线观看宜春院 | 在线观看精品视频 | 97精品国产aⅴ | 综合久久五月天 | 日韩网站免费观看 | 国产精品99久久久久的智能播放 | 人人看人人爱 | 天堂av免费看 | 国产亚洲情侣一区二区无 | 91九色自拍 | 欧美极度另类性三渗透 | 激情综合网婷婷 | 久久九九精品 | 黄色在线免费观看网站 | 91麻豆网站| 国产很黄很色的视频 | 久久久久看片 | 久久噜噜少妇网站 | 成人在线观看影院 | 国产一级免费片 | 国产精品一区二区免费 | 中文在线a在线 | 99亚洲精品视频 | 在线99 | 精品美女久久久久久免费 | 国产麻豆精品95视频 | 玖玖视频网 | 色欧美成人精品a∨在线观看 | 最近中文字幕视频完整版 | 久久免费视频在线观看 | 97精品一区| 又黄又爽又无遮挡的视频 | 日韩精品在线看 | 亚洲激情五月 | 91丨九色丨国产丨porny精品 | 一级免费av| 久久国产午夜精品理论片最新版本 | 91av免费看 | 久久黄色a级片 | 久久久久久激情 | 日韩欧美精品一区 | 亚洲精品影院在线观看 | 天天操网站 | 国产一区电影在线观看 | 国产成人久久 | 综合网伊人 | 精品国产理论片 | 欧美日韩另类在线 | 五月开心网 | 二区中文字幕 | 国产黄色视 | 久久av福利 | 久久在线看 | 男女日麻批 | 蜜桃av综合网 | 韩日三级在线 | av在线在线 | 国产成人久久精品亚洲 | 久久午夜免费观看 | 久久精品中文字幕少妇 | 亚洲aⅴ久久精品 | 一区二区三区播放 | 国产成人三级在线 | 色婷婷综合在线 | 免费av网址大全 | www.天天干 | 色视频网站在线观看一=区 a视频免费在线观看 | 菠萝菠萝蜜在线播放 | 久久综合狠狠综合久久激情 | 日韩欧美一区二区三区在线观看 | 午夜精品视频一区二区三区在线看 | 欧美综合在线视频 | 成人在线视频免费看 | 狠狠操天天射 | 中文字幕在线日亚洲9 | avav99| 天天射天天爽 | 2023年中文无字幕文字 | 五月天狠狠操 | 亚洲精品美女免费 | 在线网站黄| 九精品| 免费国产在线视频 | 欧美日韩高清在线观看 | 国产精品毛片久久久久久久久久99999999 | 91在线视频免费观看 | 午夜视频在线网站 | 91高清在线看 | 九九热视频在线免费观看 | 亚洲视频久久久 | 欧美日韩国产精品一区二区亚洲 | 亚洲精品欧美精品 | 成人va在线观看 | 国产中文字幕一区二区三区 | 337p西西人体大胆瓣开下部 | 国产黄色片网站 | 新版资源中文在线观看 | 日本久久久久久科技有限公司 | 国产午夜精品免费一区二区三区视频 | 天天干天天插伊人网 | 国产日韩精品在线观看 | 国产精品久久久久久久久久新婚 | 日日日日日 | 久久综合婷婷国产二区高清 | 亚洲视频h| 久久久久五月天 | 粉嫩av一区二区三区四区 | 日韩视频免费看 | 在线一二三区 | 丁香视频五月 | 2020天天干天天操 | 91精品久久久久久粉嫩 | 97人人网 | 国产免费嫩草影院 | av大全在线看| 国产一级高清视频 | 天天躁日日躁狠狠躁 | 欧美日韩一区二区三区在线免费观看 | 久久久精华网 | 中文字幕免费观看视频 | 久久精品播放 | 免费日韩一区二区三区 | av一区二区在线观看中文字幕 | 久久第四色| 三级a毛片 | 久久欧美在线电影 | 国产一区成人在线 | 中文字幕中文字幕 | 99久久婷婷国产综合亚洲 | 国产免费二区 | 五月天免费网站 | 日韩久久激情 | 午夜久久网站 | 成人精品一区二区三区电影免费 | 日日爽视频 | 能在线观看的日韩av | 日韩一区二区久久 | 在线色资源 | 国产精品欧美一区二区三区不卡 | 69热国产视频 | 少妇视频在线播放 | 在线观看日韩免费视频 | 国产香蕉在线 | 中文在线8资源库 | 天天躁天天躁天天躁婷 | 日韩在线高清视频 | 亚洲综合婷婷 | 成人免费视频免费观看 | 91少妇精拍在线播放 | 欧美一二三区在线观看 | 亚洲国产精品成人综合 | 国产一区视频免费在线观看 | 国产精品成人品 | 日本天天操 | 精品 一区 在线 | 日韩亚洲在线视频 | 国产1区2区 | www好男人 | 成 人 黄 色 视频 免费观看 | 久久久久久久久久久久国产精品 | 日韩av资源站 | 国产精品免费看久久久8精臀av | 天天操夜夜爱 | 91麻豆精品国产91久久久久久 | 久久综合精品国产一区二区三区 | 在线精品在线 | 日韩久久久久久久久久久久 | 综合网成人 | 久久九九久久九九 | 丁香六月婷 | 欧美精品久久久久 | 久久一区二区三区国产精品 | 四虎影视精品成人 | 免费看的黄网站 | 91黄在线看 | 波多野结衣视频一区 | 亚洲色图色| 九九九在线 | 麻豆视频国产精品 | 久久99精品久久久久久清纯直播 | 日韩欧美在线中文字幕 | 免费97视频| 久久99精品一区二区三区三区 | 黄色毛片在线 | 亚洲国产精品久久久久久 | 久久这里| 久久久久免费网 | 五月婷综合网 | 久久情侣偷拍 | 丁香激情综合国产 | 开心激情综合网 | 国产精品都在这里 | 国产欧美精品一区二区三区 | 在线亚洲欧美视频 | 99久久久国产精品 | 韩日精品在线 | 色99视频| 国产精品久久久久久久av电影 | 国产一级久久久 | 国产精国产精品 | 亚洲精品免费观看视频 | 日日夜夜免费精品 | 精品久久国产 | 麻豆高清免费国产一区 | 国产精品一区在线播放 | 欧美日韩另类视频 | 成人资源网 | 亚洲精品乱码久久久久久蜜桃欧美 | 日韩欧美aaa | 欧美日韩亚洲在线 | 欧美另类69| 伊人伊成久久人综合网小说 | 天天操天天干天天综合网 | 免费在线成人av电影 | 色噜噜日韩精品一区二区三区视频 | 在线观看 国产 | 国产中文字幕免费 | 久草| 欧美极品少妇xbxb性爽爽视频 | 国产精品久久久久久久久久免费看 | 亚洲黄色免费网站 | 2024av| 91精品一 | 国产破处精品 | 波多野结衣久久资源 | 91麻豆视频网站 | 久久综合国产伦精品免费 | 在线观看午夜 | 国产在线视频不卡 | 日韩在线视频一区二区三区 | 欧美大片在线看免费观看 | 欧美日韩后 | 久久精品日产第一区二区三区乱码 | 中文字幕日本电影 | 欧美成年人在线视频 | 在线观看视频一区二区三区 | 久久av免费电影 | 91成版人在线观看入口 | 久操伊人 | 91av网址| 久久国产午夜精品理论片最新版本 | 视频国产区 | 欧美日韩精品网站 | wwwwwww黄 | 色夜视频 | 91免费试看 | 国产专区在线播放 | 久精品视频在线 | 特级黄色片免费看 | 97国产一区 | av中文字幕在线观看网站 | 香蕉视频在线视频 | 日韩av成人在线观看 | 免费www视频 | 国产精选在线观看 | 久热电影 | 久久激情久久 | 综合久久2023 | 婷婷久久综合九色综合 | 四虎影院在线观看av | 日日夜夜精品视频 | 69欧美视频 | 欧美影院久久 | 香蕉网址 | 亚洲一区精品二人人爽久久 | 99久久精品免费看国产四区 | 天天玩天天干天天操 | 99国产成+人+综合+亚洲 欧美 | av软件在线观看 | 一区二区三区电影大全 | 国产日韩高清在线 | 一区国产精品 | 欧美国产日韩一区二区三区 | 国产在线视频在线观看 | 91视频在线 | 久久精品伊人 | 天天艹日日干 | 骄小bbw搡bbbb揉bbbb | 亚洲香蕉在线观看 | 97涩涩视频 | 亚洲精品乱码久久久久久蜜桃欧美 | 中文字幕在线视频免费播放 | 日本三级不卡视频 | 岛国av在线 | 久久免费视频这里只有精品 | 国产精品久久片 | 久久高清视频免费 | 国产在线精品一区 | 99久久久国产精品美女 | 99中文字幕| 在线观看视频精品 | 在线a视频免费观看 | 免费成视频 | 国产精品乱码一区二区视频 | 欧美日韩一区久久 | 看黄色.com| 久久久久久久久久久综合 | 国产精品久久嫩一区二区免费 | 婷婷资源站 | 在线 国产 日韩 | 一区二区中文字幕在线 | 91av在线看| 亚洲男男gaygay无套同网址 | 五月亚洲 | 日本精品一区二区在线观看 | 激情偷乱人伦小说视频在线观看 | 久久综合亚洲鲁鲁五月久久 | 超级av在线 | 91久久久久久久 | 国产日韩亚洲 | 特级a毛片 | 色九九在线 | 国产精品久久精品 | 丁香婷婷久久 | 成全免费观看视频 | 国产一区在线免费 | 国产色道| 一区二区三区在线观看免费视频 | 91av播放 | 成人免费观看完整版电影 | 91在线永久| 337p日本大胆噜噜噜噜 | 日本中文字幕在线播放 | 国产美女网 | adn—256中文在线观看 | 免费成人av| 日韩精品视频免费在线观看 | 久久久91精品国产一区二区精品 | 中文字幕在线影视资源 | 国产精品免费一区二区 | 91麻豆传媒 | 亚洲,播放 | 免费福利小视频 | 久久久国产精品电影 | 亚洲狠狠操 | 特级毛片在线免费观看 | 国产成人免费网站 | 美女视频一区二区 | 在线播放视频一区 | 国产99在线免费 | 91九色在线视频 | 正在播放国产精品 | 香蕉影院在线播放 | 色资源中文字幕 | 亚洲成a人片77777潘金莲 | 精品久久久久久久久久久久久久久久 | 高清一区二区 | 久久婷五月 | 国产精品手机在线 | 成人黄色小说视频 | 精品中文字幕视频 | 九九热视频在线免费观看 | 亚洲在线观看av | 成人免费观看网站 | 天天人人综合 | 在线观看日韩免费视频 | 久久精品黄| 亚洲电影影音先锋 | 中文字幕91在线 | 国产精品福利在线观看 | 久久第四色 | 亚州国产精品久久久 | 亚洲成av人片在线观看www | 国产精品久久久久久久久久ktv | 中文字幕 在线 一 二 | 国产精品久久久久久模特 | 日韩专区av| 韩国av免费 | 6080yy午夜一二三区久久 | 久久久久女教师免费一区 | 九九热在线播放 | 国产精品观看在线亚洲人成网 | 91av99| 欧美精品一区二区在线观看 | 国产视 | 国产91欧美 | 米奇狠狠狠888 | 99久久综合精品五月天 | 狠狠的操| 一区二区三区 中文字幕 | 在线国产福利 | 97人人澡人人爽人人模亚洲 | 91漂亮少妇露脸在线播放 | 99精品在线观看 | 精品国产美女在线 | 九九免费观看全部免费视频 | 超级碰碰碰碰 | 婷婷五综合 | 美女网站视频色 | 亚洲 av网站 | 国产伦精品一区二区三区在线 | 欧美另类z0zx | 高清av免费看 | 亚洲成人家庭影院 | 国产日韩欧美在线免费观看 | 在线a视频免费观看 | 久草在线在线 | 日韩一级电影在线观看 | 国偷自产视频一区二区久 | 久久精品国产美女 | 日韩激情精品 | 天天爽天天爽 | aaa日本高清在线播放免费观看 | 国产在线超碰 | 国产成人在线免费观看 | 亚洲精品在线免费观看视频 | 国产男男gay做爰 | 91九色在线视频观看 | 激情av在线资源 | 免费观看成人网 | 久久久久色 | 成人黄色小视频 | 久久国产视频网站 | 一本一本久久a久久精品综合妖精 | 亚州av一区 | 69xxxx欧美| 久久公开免费视频 | 国内精品久久久久久久影视麻豆 | 国产视频九色蝌蚪 | 一区二区电影网 | 天天操夜夜摸 | 福利视频在线看 | 久久国产精品系列 | 精品国产三级a∨在线欧美 免费一级片在线观看 | 国产成人精品aaa | 国产精品毛片久久久久久 | 亚洲欧美日本A∨在线观看 青青河边草观看完整版高清 | 五月婷婷丁香综合 | 日韩专区在线 | 欧美动漫一区二区三区 | 亚洲女人天堂成人av在线 | 亚洲三级国产 | 亚洲成aⅴ人片久久青草影院 | 久久久免费观看 | 国内精品视频在线 | 99久久99久久精品国产片果冰 | 韩国视频一区二区三区 | 91大神免费在线观看 | 国产一区二区在线免费播放 | 精品一区免费 | 久久9精品| 免费日韩一区二区三区 | 国产精品视频永久免费播放 | 91九色国产蝌蚪 | 亚洲在线看 | 三级黄色网址 | 五月婷影院 | 天天色宗合 | 国产在线观看你懂得 | 中文字幕精品久久 | 国产精品久久久久久久7电影 | 国产高清日韩欧美 | 91av原创| 欧美一区二区三区不卡 | 岛国av在线| 91九色精品国产 | 国产一在线精品一区在线观看 | 久久精选| 毛片一区二区 | 国产精品手机在线 | 爱爱av在线 | 久久99视频精品 | 不卡国产视频 | 96精品高清视频在线观看软件特色 | 欧美视频99 | 免费看黄色小说的网站 | 成人免费观看网站 | 久久乐九色婷婷综合色狠狠182 | 99高清视频有精品视频 | 色香天天 | 在线免费国产视频 | 在线观看mv的中文字幕网站 | 国精产品一二三线999 | 久久不射电影网 | 久久人人爽人人爽 | 国产激情免费 | 亚洲精品久久久久58 | 正在播放亚洲精品 | 国产精品美女在线观看 | 99精品欧美一区二区 | 国产精品久久久久aaaa | 91视频观看免费 | 色婷婷视频 | 久久激情视频 久久 | 色网av | 久久久久久国产精品久久 | av免费成人| 久久狠狠婷婷 | 国产麻豆精品久久 | 最新中文字幕在线资源 | 久久九九免费视频 | 黄色高清视频在线观看 | 亚洲人成免费 | 日本久久影视 | 久久久视频在线 | 中文字幕在线播放视频 | 少妇bbr搡bbb搡bbb| 狂野欧美激情性xxxx | 久久免费av电影 | 色在线高清 | 亚洲一区网 | 国产精品手机在线观看 | 国产999精品久久久久久麻豆 | 国产成人高清在线 | 91麻豆精品国产91久久久无需广告 | 在线视频18在线视频4k | 久草视频免费在线观看 | 免费情缘 | 又色又爽又激情的59视频 | 91视频午夜 | 亚洲精选在线 | 久久精品在线 | 国产精品毛片久久久 | 欧美一二三四在线 | 五月丁色 | 国产亚洲精品无 | 久久理论电影网 | 国产永久免费观看 | 日韩色一区二区三区 | 精品国产一区二区三区在线观看 | 亚洲国产一二三 | av不卡中文| 怡春院av| 在线免费中文字幕 | www.啪啪.com| 国产白浆在线观看 | 天天干人人 | 国产精品国产三级国产aⅴ9色 | 日本黄区免费视频观看 | 久久久国产毛片 | 天天射日 | 夜夜操狠狠操 | 婷婷色中文 | 国产韩国日本高清视频 | 五月天亚洲婷婷 | 黄色在线成人 | 久久久久久久av麻豆果冻 | 国产成人一级 | 欧美精品做受xxx性少妇 | 午夜精品一区二区三区在线 | 日日精品 | 蜜桃av久久久亚洲精品 | 99热官网| 99视频在线观看视频 | 成人国产精品电影 | 在线视频 区 | 一区二区三区免费在线观看视频 | 欧美日韩在线看 | 日韩精品免费一区二区 | 天天干天天射天天爽 | 亚洲影视九九影院在线观看 | 视频在线99 | 高清av中文在线字幕观看1 | 国产美腿白丝袜足在线av | 久久精品79国产精品 | 日韩免费一区二区 | 国内久久精品视频 | 美女精品久久久 | 国产一级二级在线播放 | 久久综合九色综合欧美狠狠 | 亚洲精品黄色片 | 免费视频一二三区 | 福利一区二区 | 国产黄网站在线观看 | 亚洲欧美日韩国产一区二区 | 丁香激情五月婷婷 | 国产又粗又猛又爽又黄的视频先 | 久久国产香蕉视频 | 国产精品第一页在线 | 欧美极度另类 | 午夜免费在线观看 | jizz欧美性9| 精品国产电影 | 日韩一区二区三区免费电影 | 久草视频在线免费播放 | 欧美日韩高清 | 国产福利91精品 | 玖玖综合网 | 999色视频| 国产精品中文字幕在线播放 | 欧美 日韩 视频 | 中文字幕美女免费在线 | 国产黄色大片免费看 | 日韩久久视频 | 五月婷婷综合久久 | 精品国产精品久久 | 欧美大香线蕉线伊人久久 | 99免费在线 | 日本爱爱免费 | 97在线观看免费视频 | 99电影| 日本黄色一级电影 | 91在线免费公开视频 | 日韩av在线一区二区 | 97超碰在线久草超碰在线观看 | 久草干 | 久久精品久久精品 | 特黄特色特刺激视频免费播放 | 999国产 | 91九色蝌蚪| 婷婷六月综合亚洲 | 国产精品第十页 | 在线视频app | 亚洲一区二区视频在线播放 | 欧美性色综合网 | 97精品国产97久久久久久春色 | 波多野结衣精品在线 | 亚洲成人国产 | 婷婷色资源 | 国产日韩欧美在线免费观看 | 狠狠色狠狠综合久久 | 成人一区影院 | 日韩精品视频在线免费观看 | 午夜精品久久一牛影视 | 久久久官网| 国产h片在线观看 | 国产69精品久久久久99尤 | 国产成人精品久久久久蜜臀 | 伊人久久在线观看 | 成人av中文字幕在线观看 | 日日操操 | 人人爽人人爱 | 999久久| 午夜视频一区二区三区 | 国产97碰免费视频 | 日韩av一区二区在线 | 精品999在线观看 | 国产精品一区二区三区视频免费 | 在线午夜| 97超碰网| 永久免费毛片 | 中文字幕一区二区三区在线播放 | 2021久久| 日韩影视在线观看 | 伊人久操 | 色婷婷综合久久久久中文字幕1 | 亚洲黄色大片 | 中文字幕永久免费 | 国产精品久久久一区二区三区网站 | 黄色一级免费 | 国产在线999 | 五月天综合网站 | 激情av资源 | 99自拍视频在线观看 | 亚洲午夜精 | 日本午夜免费福利视频 | 丰满少妇在线 | 精品亚洲免a | 精品免费一区 | 日本黄区免费视频观看 | 久久久99精品免费观看乱色 | 色av色av色av | 久久草视频 | 色网站在线看 | 欧美日韩国产精品一区二区亚洲 | 亚洲片在线资源 | 麻豆成人在线观看 | 亚洲日本国产精品 | 日本在线观看一区二区 | 91在线看片 | 欧美精品第一 | 六月丁香久久 | 在线欧美小视频 | 99在线热播 | 成人黄色毛片 | 国产精品久久二区 | 中文字幕黄色网址 | 国产视频在线免费观看 | 色丁香综合| 久久精品老司机 | 国产美女视频黄a视频免费 久久综合九色欧美综合狠狠 | 96精品在线 | 成人永久免费 | 在线电影91 | 超碰人人干人人 | 久久综合狠狠综合久久狠狠色综合 | 中文字幕在线免费看 | 国产精品精品久久久久久 | 国内免费久久久久久久久久久 | av中文字幕在线观看网站 | 国产精品综合久久久 | 一区二区精品视频 | 午夜免费视频网站 | 中文一二区 | 国产人成精品一区二区三 | 色com网| 一级片色播影院 | 日韩成人在线一区二区 | 国产一区在线免费观看 | 中文字幕免费在线 | 又色又爽又黄高潮的免费视频 | 精品一区二区影视 | 国产.精品.日韩.另类.中文.在线.播放 | 国产麻豆精品一区二区 | 丰满少妇高潮在线观看 | 日韩在线不卡 | 看片在线亚洲 | 中文免费 | 黄色动态图xx |