轻量UI库-Masonry解析-链式语法

2018-10-17  本文已影响1人  superKelly

看一段典型的Masonry布局代码

[label1 mas_makeConstraints:^(MASConstraintMaker *make) {
        make.top.equalTo(superview.mas_top).with.offset(padding.top); //with is an optional semantic filler
        make.left.equalTo(superview.mas_left).with.offset(padding.left);
        make.bottom.equalTo(superview.mas_bottom).with.offset(-padding.bottom);
        make.right.equalTo(superview.mas_right).with.offset(-padding.right);
    }];

简约的将top,left,bottom等布局因子用链式语法来设置。这篇文章就来解析一下如何实现链式语法。


keypoint one

点语法在oc中并不是直接调用成员变量,而是调用getter/setter函数,由getter函数来访问并返回成员变量。为了节省代码量,oc推出property来自动生成以下划线开头的实例变量_xx和对应的getter和setter方法。我们看看在property出现之前,点语法、getter/setter和带下划线成员变量之间的关系:

KLabel.h
@interface KLabel : UILabel
{
    NSString *_member;
}
- (NSString*)member;
-(void)setMember:(NSString*)newMember;
@end

KLabel.m
@implementation KLabel
-(void)setMember:(NSString*)newMember
{
    if(newMember!=_member)
    {
        _member=[newMember copy];
    }
}
- (NSString*)member
{
    return _member;
}
@end

ViewController.m
- (void)viewDidLoad {
    [super viewDidLoad];
    KLabel *a = [[KLabel alloc] init];
    a.member = @"hello";
    NSLog(a.member);
}

结果:

2018-10-17 09:53:57.397540+0800 Demo2[826:1575293] call setter function setMember
2018-10-17 09:53:57.397727+0800 Demo2[826:1575293] call getter function member
2018-10-17 09:53:57.397836+0800 Demo2[826:1575293] hello

总结:可见,实际上并没有member这一成员变量,member是函数,真正的成员变量是_member。


keypoint two

实现链式调用
当function返回值为对象时,就能继续执行点式调用。

@interface KLabel : UILabel
- (KLabel*)func1a;
- (KLabel*)func1b;
@end

@implementation KLabel
- (KLabel*)func1a
{
    NSLog(@"func1a");
    return self;
}
- (KLabel*)func1b
{
    NSLog(@"func1b");
    return self;
}
@end

ViewController.m
- (void)viewDidLoad {
    [super viewDidLoad];
    KLabel *a = [[KLabel alloc] init];
    a.func1a.func1b;
}

结果:

2018-10-17 10:12:26.633590+0800 Demo2[1564:1590405] func1a
2018-10-17 10:12:26.633771+0800 Demo2[1564:1590405] func1b

keypoint three:

此时还没传入参数。想传入参数,将返回值self改为block,则能用(@"a")的方式来传参并执行。block的返回值设为self,继续实现链式调用的功能。

- (KLabel* (^)(NSString* c))func1c
{
    KLabel* (^myBlock)(NSString* c);
    myBlock = ^(NSString* c){
        NSLog(@"func1c %@",c);
        return self;
    };
    return myBlock;
}
- (KLabel* (^)(NSString* d))func1d
{
    KLabel* (^myBlock)(NSString* d);
    myBlock = ^(NSString* d){
        NSLog(@"func1d %@",d);
        return self;
    };
    return myBlock;
}
ViewController.m
- (void)viewDidLoad {
    [super viewDidLoad];
    KLabel *a = [[KLabel alloc] init];
    a.func1c(@"c").func1d(@"d");
}

结果:

2018-10-17 10:28:03.298684+0800 Demo2[2184:1603207] func1c c
2018-10-17 10:28:03.298877+0800 Demo2[2184:1603207] func1d d

a.func1c -- 返回block:KLabel* (^myBlock)(NSString* d)
a.func1c(@"c")--执行block,并传入参数@“c”,block返回值为self,a.func1c(@"c").func1d(@"d")即转为self.func1d(@"d")

上一篇下一篇

猜你喜欢

热点阅读