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;
-
ivar_getName(Ivar _Nonnull v)
獲取ivar的名字。 -
ivar_getOffset(Ivar _Nonnull v)
獲取ivar的偏移值。 -
ivar_getTypeEncoding(Ivar _Nonnull v)
獲取ivar的類型編碼。 -
YYEncodingType YYEncodingGetType(const char *typeEncoding)
獲取類型的枚舉。
4. YYClassMethodInfo
根據方法結構體創建一個類方法實例。
- (instancetype)initWithMethod:(Method)method;
-
method_getName(Method _Nonnull m)
獲取方法選擇器。 -
method_getImplementation(Method _Nonnull m)
獲取方法的實現。 -
sel_getName(SEL _Nonnull sel)
獲取方法的名稱。 -
method_getTypeEncoding(Method _Nonnull m)
獲取方法的類型編碼,為字符串類型。 -
method_copyReturnType(Method _Nonnull m)
獲取方法的返回類型編碼,為字符串類型。 -
method_getNumberOfArguments(Method _Nonnull m)
獲取入參的個數。 -
method_copyArgumentType(Method _Nonnull m, unsigned int index)
獲取方法參數的入參類型。
5. YYClassPropertyInfo
根據屬性創建屬性實例。
- (instancetype)initWithProperty:(objc_property_t)property;
-
property_getName(objc_property_t _Nonnull property)
獲取屬性的名稱。 -
property_copyAttributeList(objc_property_t _Nonnull property, unsigned int * _Nullable outCount)
獲取屬性列表。 -
FOUNDATION_EXPORT SEL NSSelectorFromString(NSString *aSelectorName);
根據字符串返回方法選擇器。
6. YYClassInfo
根據類或類名字創建類實例。
+ (nullable instancetype)classInfoWithClass:(Class)cls;
+ (nullable instancetype)classInfoWithClassName:(NSString *)className;
-
const void *CFDictionaryGetValue(CFDictionaryRef theDict, const void *key);
返回key指定的類。 -
class_copyMethodList(Class _Nullable cls, unsigned int * _Nullable outCount)
獲取方法列表。 -
class_copyPropertyList(Class _Nullable cls, unsigned int * _Nullable outCount)
獲取屬性列表。 -
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