NSObject+YYModel的學習

圖片來之網絡

Runtime 很神奇,很有魔力
Runtime 在代碼開發中一直沒有用到過
那只是因為我們寫的是業務代碼 !_!

今天看到NSObject+YYModel時,仿佛又一次在學習Runtime了,似曾相識。其中用到的YYClassInfo類,對Model轉化需要用到的所有信息進行了處理,看到的就是runtime定義的哪些屬性。

YYClassInfo

1. 類型
/**
 Type encoding's type.
 */
typedef NS_OPTIONS(NSUInteger, YYEncodingType) {
    YYEncodingTypeMask       = 0xFF, ///< mask of type value
    YYEncodingTypeUnknown    = 0, ///< unknown
    YYEncodingTypeVoid       = 1, ///< void
    YYEncodingTypeBool       = 2, ///< bool
    YYEncodingTypeInt8       = 3, ///< char / BOOL
    YYEncodingTypeUInt8      = 4, ///< unsigned char
    YYEncodingTypeInt16      = 5, ///< short
    YYEncodingTypeUInt16     = 6, ///< unsigned short
    YYEncodingTypeInt32      = 7, ///< int
    YYEncodingTypeUInt32     = 8, ///< unsigned int
    YYEncodingTypeInt64      = 9, ///< long long
    YYEncodingTypeUInt64     = 10, ///< unsigned long long
    YYEncodingTypeFloat      = 11, ///< float
    YYEncodingTypeDouble     = 12, ///< double
    YYEncodingTypeLongDouble = 13, ///< long double
    YYEncodingTypeObject     = 14, ///< id
    YYEncodingTypeClass      = 15, ///< Class
    YYEncodingTypeSEL        = 16, ///< SEL
    YYEncodingTypeBlock      = 17, ///< block
    YYEncodingTypePointer    = 18, ///< void*
    YYEncodingTypeStruct     = 19, ///< struct
    YYEncodingTypeUnion      = 20, ///< union
    YYEncodingTypeCString    = 21, ///< char*
    YYEncodingTypeCArray     = 22, ///< char[10] (for example)
    
    YYEncodingTypeQualifierMask   = 0xFF00,   ///< mask of qualifier
    YYEncodingTypeQualifierConst  = 1 << 8,  ///< const
    YYEncodingTypeQualifierIn     = 1 << 9,  ///< in
    YYEncodingTypeQualifierInout  = 1 << 10, ///< inout
    YYEncodingTypeQualifierOut    = 1 << 11, ///< out
    YYEncodingTypeQualifierBycopy = 1 << 12, ///< bycopy
    YYEncodingTypeQualifierByref  = 1 << 13, ///< byref
    YYEncodingTypeQualifierOneway = 1 << 14, ///< oneway
    
    YYEncodingTypePropertyMask         = 0xFF0000, ///< mask of property
    YYEncodingTypePropertyReadonly     = 1 << 16, ///< readonly
    YYEncodingTypePropertyCopy         = 1 << 17, ///< copy
    YYEncodingTypePropertyRetain       = 1 << 18, ///< retain
    YYEncodingTypePropertyNonatomic    = 1 << 19, ///< nonatomic
    YYEncodingTypePropertyWeak         = 1 << 20, ///< weak
    YYEncodingTypePropertyCustomGetter = 1 << 21, ///< getter=
    YYEncodingTypePropertyCustomSetter = 1 << 22, ///< setter=
    YYEncodingTypePropertyDynamic      = 1 << 23, ///< @dynamic
};

將基礎數據類型及類的類型,遠程對象的消息傳遞的類型和屬性類型進行枚舉。其中遠程對象的消息傳遞類型用的比較少,用到的關鍵字:

  • in:參數是輸入參數;
  • out:參數是輸出參數;
  • inout:參數即是輸入參數,又是輸出參數;
  • bycopy:復制傳值;
  • byref:引用傳值;
  • oneway:方法是異步的,也就是函數調用會立即返回(否則的話,調用者會一直堵塞,直到被調用函數執行完畢,即使被調用者返回值是void,也同樣會被阻塞)。它的返回值必須是void(其它返回值是沒有意義的,因為被調用函數是立即返回,必然無法得到正確的返回值)。
2. 編碼含義

Objective-C 類型編碼

編碼 含義
c 一個字符
i 一個整型
s 一個短整型
l 一個長整型, 在64位機器上是32位長度
q 一個長長整型
C 一個無符號字符
I 一個無符號整型
S 一個無符號短整型
L 一個無符號長整型
Q 一個無符號長長整型
f 一個浮點型單精度數
d 一個浮點型雙精度數
B 一個布爾型
v 一個void類型
* 一個字符串(char *)
@ 一個對象(靜態類型或id類型)
# 一個類對象class
一個方法選擇器SEL
[array type] 一個數組類型
{name=type...} 一個結構體類型
(name=type...) 一個union類型
bnum 一個數字位的位域
^type 一個指針類型
一個未知類型(通常這個編碼被用于函數指針)

Objective-C 方法編碼

編碼 含義
r const
n in
N inout
o out
O bycopy
R byref
V oneway

屬性類型字符串

編碼 含義
R readonly
C copy
& retain
N nonatomic
G<name> 自定義Getter方法
S<name> 自定義Setter方法
D @dynamic 動態的屬性
W __weak 屬性
P 垃圾回收
t<encoding> 使用舊樣式編碼的特殊類型

根據typeEncoding字符串獲取類型。

YYEncodingType YYEncodingGetType(const char *typeEncoding);
3. YYClassIvarInfo

根據ivar結構體創建實例變量,如果發生錯誤則返回nil。

- (instancetype)initWithIvar:(Ivar)ivar;
  1. ivar_getName(Ivar _Nonnull v) 獲取ivar的名字。
  2. ivar_getOffset(Ivar _Nonnull v) 獲取ivar的偏移值。
  3. ivar_getTypeEncoding(Ivar _Nonnull v) 獲取ivar的類型編碼。
  4. YYEncodingType YYEncodingGetType(const char *typeEncoding) 獲取類型的枚舉。
4. YYClassMethodInfo

根據方法結構體創建一個類方法實例。

- (instancetype)initWithMethod:(Method)method;
  1. method_getName(Method _Nonnull m) 獲取方法選擇器。
  2. method_getImplementation(Method _Nonnull m) 獲取方法的實現。
  3. sel_getName(SEL _Nonnull sel) 獲取方法的名稱。
  4. method_getTypeEncoding(Method _Nonnull m) 獲取方法的類型編碼,為字符串類型。
  5. method_copyReturnType(Method _Nonnull m) 獲取方法的返回類型編碼,為字符串類型。
  6. method_getNumberOfArguments(Method _Nonnull m) 獲取入參的個數。
  7. method_copyArgumentType(Method _Nonnull m, unsigned int index) 獲取方法參數的入參類型。
5. YYClassPropertyInfo

根據屬性創建屬性實例。

- (instancetype)initWithProperty:(objc_property_t)property;
  1. property_getName(objc_property_t _Nonnull property) 獲取屬性的名稱。
  2. property_copyAttributeList(objc_property_t _Nonnull property, unsigned int * _Nullable outCount) 獲取屬性列表。
  3. FOUNDATION_EXPORT SEL NSSelectorFromString(NSString *aSelectorName); 根據字符串返回方法選擇器。
6. YYClassInfo

根據類或類名字創建類實例。

+ (nullable instancetype)classInfoWithClass:(Class)cls;
+ (nullable instancetype)classInfoWithClassName:(NSString *)className;
  1. const void *CFDictionaryGetValue(CFDictionaryRef theDict, const void *key); 返回key指定的類。
  2. class_copyMethodList(Class _Nullable cls, unsigned int * _Nullable outCount) 獲取方法列表。
  3. class_copyPropertyList(Class _Nullable cls, unsigned int * _Nullable outCount) 獲取屬性列表。
  4. class_copyIvarList(Class _Nullable cls, unsigned int * _Nullable outCount) 獲取類變量列表。

YYModel

看了一圈代碼,直接看暈了。處理的內容還是比較多啊,用的還都不是NS層面的,使用runtime的方法比較多。

1. YYEncodingNSType

將所有的NS基礎類類型做了一個枚舉。

typedef NS_ENUM (NSUInteger, YYEncodingNSType) {
    YYEncodingTypeNSUnknown = 0,
    YYEncodingTypeNSString,
    YYEncodingTypeNSMutableString,
    YYEncodingTypeNSValue,
    YYEncodingTypeNSNumber,
    YYEncodingTypeNSDecimalNumber,
    YYEncodingTypeNSData,
    YYEncodingTypeNSMutableData,
    YYEncodingTypeNSDate,
    YYEncodingTypeNSURL,
    YYEncodingTypeNSArray,
    YYEncodingTypeNSMutableArray,
    YYEncodingTypeNSDictionary,
    YYEncodingTypeNSMutableDictionary,
    YYEncodingTypeNSSet,
    YYEncodingTypeNSMutableSet,
};
2. 轉化為NSNumber

static force_inline NSNumber *YYNSNumberCreateFromID(__unsafe_unretained id value) ;將id類型轉化為NSNumber類型.

3. 解析string為date

static force_inline NSDate *YYNSDateFromString(__unsafe_unretained NSString *string)輸入任意類型的時間格式字符串轉化為NSDate。
static YYNSDateParseBlock blocks[kParserNum + 1] = {0};按字符串的長度來判斷是屬于哪一種格式的時間格式,創建對應的格式化類型。

4. 類型的賦值和獲取

整個Model的主要功能就是根據Model中定義的屬性類型,將JSON中的值進行轉化為這個類型,然后賦值給這個屬性。使用了很多CF*和Runtime的代碼。一堆的業務代碼,一堆的switch分類型處理。理解了C語言在OC的使用廣泛,指針,結構體等等用的溜溜的。
有機會再來研究下這些代碼。

// END

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容