YYModel源码解析(二)YYClassInfo类

1 . NS_ASSUME_NONNULL_BEGIN

    #define NS_ASSUME_NONNULL_BEGIN _Pragma("clang assume_nonnull begin")
    #define NS_ASSUME_NONNULL_END   _Pragma("clang assume_nonnull end")

在YYClassInfo.h的最开始便可以看到这个宏,和NS_ASSUME_NONNULL_BEGIN匹配使用的是NS_ASSUME_NONNULL_END,在这两个宏之间区域的代码的指针都可以认为是nonnull的,如果有需要申明为nullable的话,只需要去单独申明nullable的指针便可以了。

不过,为了安全起见,苹果还制定了几条规则:

typedef定义的类型的nullability特性通常依赖于上下文,即使是在Audited Regions中,也不能假定它为nonnull。
复杂的指针类型(如id *)必须显示去指定是nonnull还是nullable。例如,指定一个指向nullable对象的nonnull指针,可以使用”__nullable id * __nonnull”。
我们经常使用的NSError **通常是被假定为一个指向nullable NSError对象的nullable指针。

2 . YYEncodingType
接下来可以看到作者自定义的一大串枚举

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
};

基本把属性本身类型和修饰类型,及方法返回类型都进行了枚举,然后又通过:

YYEncodingType YYEncodingGetType(const char *typeEncoding);

把编译器编码的 Type Encodings 在运行时翻译为定义好的枚举类型,运作在YYModel体系中。
这里关于Type Encodings的资料可参考:
https://developer.apple.com/library/mac/documentation/Cocoa/Conceptual/ObjCRuntimeGuide/Articles/ocrtTypeEncodings.html
https://developer.apple.com/library/mac/documentation/Cocoa/Conceptual/ObjCRuntimeGuide/Articles/ocrtPropertyIntrospection.html

3 . YYClassInfo类
YYClassInfo 类,其实很简单,简单说就是把 Class(struct objc_class *) 封装成了类并且做了一部分扩展。
对比一下 struct objc_class 和 YYClassInfo的属性定义部分:

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;

YYClassInfo:

@property (nonatomic, assign, readonly) Class cls; ///< class object
@property (nullable, nonatomic, assign, readonly) Class superCls; ///< super class object
@property (nullable, nonatomic, assign, readonly) Class metaCls;  ///< class's meta class object
@property (nonatomic, readonly) BOOL isMeta; ///< whether this class is meta class
@property (nonatomic, strong, readonly) NSString *name; ///< class name
@property (nullable, nonatomic, strong, readonly) YYClassInfo *superClassInfo; ///< super class's class info
@property (nullable, nonatomic, strong, readonly) NSDictionary *ivarInfos; ///< ivars
@property (nullable, nonatomic, strong, readonly) NSDictionary *methodInfos; ///< methods
@property (nullable, nonatomic, strong, readonly) NSDictionary *propertyInfos; ///< properties

可以看出,相对于 struct objc_class ,YYClassInfo中多了对元类相关的一些属性,然后少了几个没什么大用的描述性属性,别的都是一一对应的。
然后 YYClassIvarInfo 、YYClassMethodInfo、YYClassPropertyInfo 其实也分别对应封装的 struct objc_ivar 、struct objc_method、 struct objc_property。

然后看一下YYClassInfo的方法:
+ (nullable instancetype)classInfoWithClass:(Class)cls;
+ (nullable instancetype)classInfoWithClassName:(NSString *)className;
没啥好说的,根据传入的参数获取生成到的对应YYClassInfo对象。

更新方法:
- (void)setNeedUpdate;
- (BOOL)needUpdate;
如果你的类有了一些改变,譬如用 'class_addMethod()' 之类的函数动态添加了方法,那么就要手动调用一下setNeedUpdate,来通知更新对应YYClassInfo缓存,之后needUpdate会返回YES, 这种情况下就需要重新用 'classInfoWithClass' 或者 'classInfoWithClassName' 更新信息。

4 . 别的

(1). 在定义 YYClassIvarInfo 中的 offset属性是用到 ptrdiff_t 类型

ptrdiff_t :ptrdiff_t是C/C++标准库中定义的一个与机器相关的数据类型。ptrdiff_t类型变量通常用来保存两个[指针]减法操作的结果。ptrdiff_t定义在stddef.h(cstddef)这个文件内。ptrdiff_t通常被定义为long int类型。ptrdiff_t定义在C99标准中。

ptrdiff_t
标准库类型(library type)ptrdiff_t 与 size_t 类型一样,ptrdiff_t 也是一种与机器相关的类型,在 cstddef 头文件中定义。size_t 是unsigned 类型,而 ptrdiff_t 则是 signed 整型。

size_t
这两种类型的差别体现了它们各自的用途:size_t 类型用于指明数组长度,它必须是一个正数;ptrdiff_t 类型则应保证足以存放同一数组中两个指针之间的差距,它有可能是负数 。

----来自 百度百科

你可能感兴趣的:(YYModel源码解析(二)YYClassInfo类)