Runtime之@dynamic关键字

讲述@dynamic之前,需要了解几个名词。最后再重点介绍@dynamic的用法

  • @property
    • 原子性
    • 存取控制
    • 内存管理
  • @synthesize

@property

使用 @property 声明的属性,可以方便快捷的为实例变量创建存取器(getter 和 setter)。默认以下划线 _ 开头。比如

@interface Person : NSObject
@property(nonatomic, readwrite, strong) NSString *name;
-(void)sayHello;
@end
    
@implementation Person
@synthesize name = _name;//默认情况,可以不写。除非想通过其他的单词(非_name)来使用
-(void)sayHello{
    NSLog(@"%@ says hello",_name);
}
@end

此时,我们可以通过点访问符 . 来给对象的name属性进行存取。

    //设置
    Person* person1 = [[Person alloc] init];
    person1.name = @"zhangsan";
    [person1 sayHello];
    //获取
    NSLog(@"person1's name is %@",[person1.name sayBye]);
原子性

​ 在上述例子中,我们看到了声明语句括号内的nonatomic关键字。具体原子性包含两个关键字:

  • atomic
    • 默认。原子的,意味着只有一个线程访问实例变量(的getter和setter)。
    • 线程安全,至少在当前的存取器上是安全的
    • 影响效率,使用较少
  • nonatomic
    • 非原子的,可以同时被多个线程访问
    • 效率高
    • 多线程下不安全,使用较多
存取控制

​ 在上述例子中,我们看到了声明语句括号内的readwrite关键字。表示属性的存取特征。存取控制具体包含以下几个:

  • readwrite
    • 默认属性,系统会自动给属性生成getter和setter存取器
  • readonly
    • 只读属性,只会生成getter;不能通过点运算符(setter)给属性赋值
内存管理

​ 在上述例子中,我们看到了声明语句括号内的strong关键字。表示属性内存管理的关键字,有以下几个:

  • assign
    • 默认,适用于值类型。如int、NSInteger、CGFloat、float等。
  • retain
    • 在setter方法中,会对传入的对象的引用计数器加1(理解ARC)。
  • strong
    • strong是iOS引入ARC之后的关键字,是retain的一个可选的替代。
  • weak
    • 与retain相比,对引入对象的引用计数器,不进行加1操作。经常用于delegate等。
  • copy
    • 与strong类似,区别是copy是创建了一个新对象。通常NSString类属性会使用copy。

@synthesize

​ @synthesize默认会给属性添加一个别名,比如上个例子中的_name 。默认情况下,对于@property的变量都会生成相关别名和存取器

@dynamic

​ 如果某属性已实现了自己的getter和setter(当然对于 readonly 的属性只有 getter ),可以通过@dynamic关键字来阻止自动生成getter和setter的覆盖。@dynamic关键字有两个作用:

  • 让编译器不要创建实现属性所用的实例变量;
  • 让编译器不要创建该属性的get和setter方法。

编译时没问题,在运行时才执行相应的方法,这就是所谓的动态绑定。

假如一个属性被声明为@dynamic var,然而没有提供对应的getter和setter方法,编译的时候不会报错,但是当程序运行到 instance.var = xxx 时,由于缺少setter方法会导致程序崩溃


@dynamic的使用

这里介绍三种

  • 通过私有变量来实现@dynamic
  • Category扩展
  • 动态解析和函数重签名

通过私有变量来实现@dynamic

通过私有变量来实现@dynamic,可以达到隐藏某些信息的目的。
例如DemoClass1中有个dynamicVar属性,我们将其设置为@dynamic,在程序运行期间,我们想通过他来存取某个私有(对象的)属性,简单起见,我们将对应的私有属性命名为privateVar。
.h 文件:

/***
 * 用来展示通过私有变量来实现@dynamic
 * 用于隐藏某些信息
 ***/
@interface DemoClass1 : NSObject
@property (nonatomic , strong) NSString* dynamicVar;
@end

.m 文件:

@interface DemoClass1()
@property (nonatomic, strong) NSString* privateVar;
@end

@implementation DemoClass1
@dynamic dynamicVar;
@synthesize privateVar = _privateVar;

- (void)setDynamicVar:(NSString *)dynamicVar{
    self.privateVar = dynamicVar;
}
-(NSString*)dynamicVar{
    return self.privateVar;
}

-(void)setPrivateVar:(NSString *)privateVar{
    _privateVar = privateVar;
}
-(NSString*)privateVar{
    if (!_privateVar) {
        _privateVar = @"This is private var";
    }
    return _privateVar;
}

方法的调用,我们通过调用dynamicVar的getter和setter,然后看下对应值

    DemoClass1* demo1 = [[DemoClass1 alloc] init];
    NSLog(@"Before - DemoClass1's dynamicVar is :%@", demo1.dynamicVar);
    demo1.dynamicVar = @"Set dynamicVar";
    NSLog(@"After - DemoClass1's dynamicVar is :%@", demo1.dynamicVar);

输出信息如下:

2019-01-25 09:19:13.253753+0800 property_dynamic[18448:7311996] Before - DemoClass1's dynamicVar is :This is private var
2019-01-25 09:19:13.253893+0800 property_dynamic[18448:7311996] After - DemoClass1's dynamicVar is :Set dynamicVar

我们可以看到,当我们访问dynamicVar的getter方法时,最终展示的私有变量privateVar。而访问setter,最终设置的也是privateVar。
通过@dynamic关键字最终实现了将私有信息暴漏的目的。

Category扩展

另一个较常用的用法就是Category,例如,我们想对UILabel进行行间距和列间距的设置,我们想通过两个属性来设置
.h 文件:

@interface UILabel (Demo2)
//行间距
@property (nonatomic, assign) CGFloat ltx_lineSpace;
//字间距
@property (nonatomic, assign) CGFloat ltx_wordSpace;
@end

.m 文件:

#import "UILabel+Demo2.h"
#import <objc/runtime.h>

NSString const *ltx_label_lineSpace = @"ltx_label_lineSpace";
NSString const *ltx_label_wordSpace = @"ltx_label_wordSpace";
@implementation UILabel (Demo2)
@dynamic ltx_lineSpace;
@dynamic ltx_wordSpace;

-(void)ltx_updateLabelLineSpace{
    NSString* text = self.text;
    NSInteger lineSpace = self.ltx_lineSpace;
    NSInteger wordSpace = self.ltx_wordSpace;
    if ([text length] > 0) {
        NSMutableAttributedString *attributedString = [[NSMutableAttributedString alloc] initWithString:text];
        if (lineSpace > 0) {
            NSMutableParagraphStyle *paragraphStyle = [[NSMutableParagraphStyle alloc] init];
            [paragraphStyle setLineSpacing:lineSpace];
            [attributedString addAttribute:NSParagraphStyleAttributeName value:paragraphStyle range:NSMakeRange(0, [text length])];
        }
        if (wordSpace > 0) {
            [attributedString addAttribute:NSKernAttributeName value:[NSNumber numberWithFloat:wordSpace] range:NSMakeRange(0, [text length])];
        }
        self.attributedText = attributedString;
    }
}


#pragma mark - Getter && Setter
-(void)setLtx_lineSpace:(CGFloat)ltx_lineSpace{
    NSNumber* number = [NSNumber numberWithFloat:ltx_lineSpace];
    objc_setAssociatedObject(self, &ltx_label_lineSpace, number, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    [self ltx_updateLabelLineSpace];
}
-(CGFloat)ltx_lineSpace{
    NSNumber* number =  objc_getAssociatedObject(self, &ltx_label_lineSpace);
    return [number floatValue];
}

-(void)setLtx_wordSpace:(CGFloat)ltx_wordSpace{
    NSNumber* number = [NSNumber numberWithFloat:ltx_wordSpace];
    objc_setAssociatedObject(self, &ltx_label_wordSpace, number, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    [self ltx_updateLabelLineSpace];
}
-(CGFloat)ltx_wordSpace{
    NSNumber* number =  objc_getAssociatedObject(self, &ltx_label_wordSpace);
    return [number floatValue];
}

方法的调用,我们通过调用ltx_lineSpaceltx_wordSpace 来对labe的列间距和字间距进行设置

    //设置UILabel的行间距和字间距
    self.label.ltx_lineSpace = 8;
    self.label.ltx_wordSpace = 4;

效果如下

动态修改UILabel的行间距和字间距.png

更多扩展,请移步参照:https://github.com/liangtongdev/LTxCategories

动态解析和函数重签名

即通过对@dynamic动态属性的getter和setter进行以下操作。

  • 动态解析
    • 重写NSObject的方法 + (BOOL)resolveInstanceMethod:(SEL)sel
    • 利用class_addMethod方法,将其他方法的实现添加给属性的getter和setter方法。
  • 函数重签名
    • 消息转发
      • 重写NSObject的方法-(id)forwardingTargetForSelector:(SEL)aSelector
      • 在方法内部,对动态属性的getter和setter方法进行转发,由其他对象来实现具体细节。
    • 方法重签名
      • 重写NSObject的方法 - (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
      • 在方法内,返回动态属性的getter和setter方法的新的方法签名
      • 重写NSObject的方法- (void)forwardInvocation:(NSInvocation *)anInvocation
      • 在方法内部,对重新签名的方法的target进行赋值,并唤醒

具体细节,请移步参照文章:Runtime之objc_msgSend执行流程


Demo

https://github.com/liangtongdev/Demo-property_dynamic

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,425评论 4 361
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,058评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,186评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,848评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,249评论 3 286
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,554评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,830评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,536评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,239评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,505评论 2 244
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,004评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,346评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,999评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,060评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,821评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,574评论 2 271
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,480评论 2 267

推荐阅读更多精彩内容