最近跟人交流時,提到一個問題,說iOS分類中不能添加屬性。這裏探討一下不能添加的緣由和添加的方法。
首先,建立一個person類,代碼以下:atom
XGPerson.hcode
#import <Foundation/Foundation.h> @interface XGPerson : NSObject /// 年齡 @property (nonatomic, copy) NSString *age; /// 性別 @property (nonatomic, copy) NSString *sex; - (void)text1; @end
XGPerson.m對象
#import "XGPerson.h" @implementation XGPerson - (void)text1 { NSLog(@"%s",__func__); } - (void)text2 { NSLog(@"%s",__func__); } @end
在控制器裏獲取並打印該類的成員變量、屬性和方法,代碼以下:ci
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event { // 獲取成員變量 unsigned int ivarCount = 0; Ivar *ivars = class_copyIvarList([XGPerson class], &ivarCount); for (int i = 0; i < ivarCount; i++) { Ivar ivar = ivars[i]; NSLog(@"第%d個成員變量:%s",i,ivar_getName(ivar)); } free(ivars); // 獲取屬性 unsigned int propertyCount = 0; objc_property_t *propertyList = class_copyPropertyList([XGPerson class], &propertyCount); for (int i = 0; i < propertyCount; i++) { objc_property_t property = propertyList[i]; NSLog(@"第%d個屬性:%s",i,property_getName(property)); } // 獲取方法列表 unsigned int methodCount = 0; Method *methods = class_copyMethodList([XGPerson class], &methodCount); for (int i = 0; i < methodCount; i++) { Method method = methods[i]; NSLog(@"第%d個方法:%s",i, sel_getName(method_getName(method))); } }
此時控制檯輸出以下:get
沒有分類時.pngit
這裏須要提出的是,平時使用@property的時候,系統會自動生成帶「_」的成員變量和該變量的setter和getter方法。也就是說,屬性至關於一個成員變量加getter和setter方法。那麼,在分類裏使用@property會是什麼樣子呢,下面來建立一個分類:
XGPerson+height.hio
#import "XGPerson.h" @interface XGPerson (height) @property (nonatomic, copy) NSString *height; @end
XGPerson+height.mevent
#import "XGPerson+height.h" #import <objc/runtime.h> @implementation XGPerson (height) @end
若是像上面同樣只在.h文件裏聲明height,那麼.m文件裏會出現兩個警告,意思是說沒有實現setter和getter方法。class
警告.pngimport
此時在控制器裏執行touchesBegan方法,控制檯輸出以下:
有分類未實現存取方法.png
能夠看到,此時person類裏並無添加帶「_」的成員變量,也沒有實現setter和getter方法,只是在屬性列表裏添加了height屬性。而且此時若是在控制器裏調用self.height,程序運行時會報錯,顯示找不到該方法。實現一下person分類裏的兩個方法:
XGPerson+height.m
#import "XGPerson+height.h" #import <objc/runtime.h> @implementation XGPerson (height) - (NSString *)height { } - (void)setHeight:(NSString *)height { } @end
此時在控制器裏執行touchesBegan方法,控制檯輸出以下:
有分類實現存取方法.png
能夠看到即便實現了setter和getter方法,也仍然沒有添加帶「」的成員變量,也就是說,在setter和getter方法裏仍然不能直接訪問如下劃線開頭的成員變量,由於在分類裏用@property聲明屬性時系統並無添加以「」開頭的成員變量。此時要達到添加的目的可使用運行時的關聯對象。示例代碼以下:
XGPerson+height.m
#import "XGPerson+height.h" #import <objc/runtime.h> @implementation XGPerson (height) - (NSString *)height { return objc_getAssociatedObject(self, @"height"); } - (void)setHeight:(NSString *)height { objc_setAssociatedObject(self, @"height", height, OBJC_ASSOCIATION_COPY_NONATOMIC); } @end
固然也能夠在setter和getter方法裏訪問該類其餘的屬性,好比在UIView的分類的裏添加x、y屬性,能夠直接返回self.frame.origin.x和self.frame.origin.y。
總結
在分類裏使用@property聲明屬性,只是將該屬性添加到該類的屬性列表,並聲明瞭setter和getter方法,可是沒有生成相應的成員變量,也沒有實現setter和getter方法。因此說分類不能添加屬性。可是在分類裏使用@property聲明屬性後,又實現了setter和getter方法,那麼在這個類之外能夠正常經過點語法給該屬性賦值和取值。就是說,在分類裏使用@property聲明屬性,又實現了setter和getter方法後,能夠認爲給這個類添加上了屬性。