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