iOS分類不能添加屬性緣由的探索

最近跟人交流時,提到一個問題,說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方法後,能夠認爲給這個類添加上了屬性。

相關文章
相關標籤/搜索