Masonry讲解与实战中内存优化

一个朋友说他原本用frame做代码布局,打算试试约束布局,好吧!懒得一一说就默默的写篇文章。。。Masonry相对于原生来说,在代码添加约束上非常强大、实用、简便(想试试原生约束可以看我之前写的一篇帖子代码约束NSLayoutConstraint)。本篇文章就简单说一下Masonry的使用以及也是重点Masonry内存优化(内存优化为后期更新),需要先将Masonry下载并导入项目中Masonry下载地址。废话就不说了,开始来点干货。

Masonry调用方法

新增约束,不管原本是否已经存在约束都新增约束,相对的也会导致增加内存的消耗。

- (NSArray *)mas_makeConstraints:(void(NS_NOESCAPE ^)(MASConstraintMaker *make))block;

更新约束

- (NSArray *)mas_updateConstraints:(void(NS_NOESCAPE ^)(MASConstraintMaker *make))block;

清除之前的约束,只保留最新约束。不管原本是否已经存在约束,都先进行清除约束,再新增约束。

- (NSArray *)mas_remakeConstraints:(void(NS_NOESCAPE ^)(MASConstraintMaker *make))block;

Masonry使用

举个栗子,用栗子来说明:

 [_logoView mas_makeConstraints:^(MASConstraintMaker *make) {
        make.top.left.mas_equalTo(0);
        make.size.mas_equalTo(CGSizeMake(MAINSIZE.width, SCR_W * 278));
 }];

block里面对需要进行约束的位置设置代码,例如make.top.left.mas_equalTo(0);它是设置logoView距离(如果没有设置视图默认为父视图)它的父视图的top、left距离为0,也就是居左、顶部,因为top、left他们距离父视图的距离相同就可以一次性的设置过去,当然上面的写法可以写成:

make.top.mas_equalTo(0);
make.left.mas_equalTo(0);

也可以写成

make.top.equalTo(self.view.mas_top);//self.view为父视图
make.left.equalTo(self.view.mas_left);

上面的写法也可以写成

make.top.equalTo(self.view.mas_top).offset(0);//self.view为父视图
make.left.equalTo(self.view.mas_left).offset(0);offset(0)//距离self.view的左边向右偏移0个单位

还可以写成用边距来设置,例如:

make.top.equalTo(self.view.mas_topMargin);
make.left.equalTo(self.view.mas_leftMargin);

这些设置都是可以的,主要是看你个人习惯。
equalTo后面带的可以是视图或者值(需要的像素),如果带的是值需要将值包装一下如:make.top.equalTo(@(0));
mas_equalTo后面带的是数值(可以是动态的数值),也只能是值。
如果你需要设置的约束是动态的,比如你确定宽度,高度不确定,那你只需要设置宽度的约束,让高度自适应就可以。反之,你确定高度不确定宽度,也是一样的道理。
注意如果你需要设置的约束为动态距离,即根据不同的屏幕间距(或边距)不同,那你设置的值不能写死,需要为动态值,跟frame设置动态距离是一样的。可以考虑用比例值,附带个宏:#define SCR_W(x) [UIScreen mainScreen].bounds.size.width/375*x。

Masonry属性

MASConstraintMaker(己身视图)
@property (nonatomic, strong, readonly) MASConstraint *left;//左边
@property (nonatomic, strong, readonly) MASConstraint *top;//上方
@property (nonatomic, strong, readonly) MASConstraint *right;//右边
@property (nonatomic, strong, readonly) MASConstraint *bottom;//下方
@property (nonatomic, strong, readonly) MASConstraint *leading;//头部
@property (nonatomic, strong, readonly) MASConstraint *trailing;//尾部
@property (nonatomic, strong, readonly) MASConstraint *width;//宽度
@property (nonatomic, strong, readonly) MASConstraint *height;//高度
@property (nonatomic, strong, readonly) MASConstraint *centerX;//横向居中,即中心点X轴坐标
@property (nonatomic, strong, readonly) MASConstraint *centerY;//纵向居中,即中心点Y轴坐标
@property (nonatomic, strong, readonly) MASConstraint *baseline;//文本基线

在iOS 8之后可以设置它的间距、边距约束

@property (nonatomic, strong, readonly) MASConstraint *leftMargin;//左边边距
@property (nonatomic, strong, readonly) MASConstraint *rightMargin;//右边边距
@property (nonatomic, strong, readonly) MASConstraint *topMargin;//上方边距
@property (nonatomic, strong, readonly) MASConstraint *bottomMargin;//下方边距
@property (nonatomic, strong, readonly) MASConstraint *leadingMargin;//头部边距
@property (nonatomic, strong, readonly) MASConstraint *trailingMargin;//尾部边距
@property (nonatomic, strong, readonly) MASConstraint *centerXWithinMargins;//X轴中心点边距
@property (nonatomic, strong, readonly) MASConstraint *centerYWithinMargins;//Y轴中心点边距

上面好像没说说道size属性,这个也是一样的既然有了width、hight,那当然也有size。

MASViewAttribute(对比视图)

与本身视图是一一对应的,只是需要添加上mas_,例如:mas_bottom

@property (nonatomic, strong, readonly) MASViewAttribute *mas_left;
@property (nonatomic, strong, readonly) MASViewAttribute *mas_top;
@property (nonatomic, strong, readonly) MASViewAttribute *mas_right;
@property (nonatomic, strong, readonly) MASViewAttribute *mas_bottom;
@property (nonatomic, strong, readonly) MASViewAttribute *mas_leading;
@property (nonatomic, strong, readonly) MASViewAttribute *mas_trailing;
@property (nonatomic, strong, readonly) MASViewAttribute *mas_width;
@property (nonatomic, strong, readonly) MASViewAttribute *mas_height;
@property (nonatomic, strong, readonly) MASViewAttribute *mas_centerX;
@property (nonatomic, strong, readonly) MASViewAttribute *mas_centerY;
@property (nonatomic, strong, readonly) MASViewAttribute *mas_baseline;
@property (nonatomic, strong, readonly) MASViewAttribute *(^mas_attribute)(NSLayoutAttribute attr);

iOS 8之后

@property (nonatomic, strong, readonly) MASViewAttribute *mas_leftMargin;
@property (nonatomic, strong, readonly) MASViewAttribute *mas_rightMargin;
@property (nonatomic, strong, readonly) MASViewAttribute *mas_topMargin;
@property (nonatomic, strong, readonly) MASViewAttribute *mas_bottomMargin;
@property (nonatomic, strong, readonly) MASViewAttribute *mas_leadingMargin;
@property (nonatomic, strong, readonly) MASViewAttribute *mas_trailingMargin;
@property (nonatomic, strong, readonly) MASViewAttribute *mas_centerXWithinMargins;
@property (nonatomic, strong, readonly) MASViewAttribute *mas_centerYWithinMargins;

本身视图属性、跟对比视图属性在苹果原生的约束布局上其实是一样的,只是masonry做了区别。

常用布局

一些常用的布局,例如:等分、优先级、自动布局、更新等。

1、等分布局

       // 水平
       [@[self.imageButton,
          self.linkButton,
          self.fontButton,
          self.sizeButton,
          self.alignmentButton,
          self.colorButton,
          self.keyboardButton] mas_distributeViewsAlongAxis:MASAxisTypeHorizontal withFixedSpacing:pading leadSpacing:pading tailSpacing:pading];
       // 高度
       [@[self.imageButton,
          self.linkButton,
          self.fontButton,
          self.sizeButton,
          self.alignmentButton,
          self.colorButton,
          self.keyboardButton] mas_makeConstraints:^(MASConstraintMaker *make) {
             make.bottom.equalTo(self).offset(-kEditorToolBarButtonBottomSpace);
             make.height.equalTo(@(kEditorToolBarButtonWidth));
         }];

均等布局mas_distributeViewsAlongAxis设置完水平布局或者垂直布局后,需要设置每一个控件自身的布局属性。

2、优先级约束

priceLabel显示的优先级比originalPriceLabel高,即先满足priceLabel

[self addSubview:self.priceLabel];
[self.priceLabel mas_makeConstraints:^(MASConstraintMaker *make) {
    make.top.equalTo(self.titleLabel.mas_bottom).mas_equalTo(4);
    make.left.equalTo(self.ebookImageView.mas_right).mas_equalTo(15);
    make.height.mas_equalTo(24);
}];

[self addSubview:self.originalPriceLabel];
[self.originalPriceLabel mas_makeConstraints:^(MASConstraintMaker *make) {
    make.centerY.equalTo(self.priceLabel);
    make.left.equalTo(self.priceLabel.mas_right).mas_equalTo(10);
    make.right.mas_lessThanOrEqualTo(self.mas_right).with.offset(-5);
}];
[self.originalPriceLabel setContentCompressionResistancePriority:(UILayoutPriorityDefaultLow) forAxis:(UILayoutConstraintAxisHorizontal)];
[self.priceLabel setContentCompressionResistancePriority:(UILayoutPriorityDefaultHigh) forAxis:(UILayoutConstraintAxisHorizontal)];

setContentCompressionResistancePriority需要与mas_lessThanOrEqualTo配套使用才能起到相应的效果。

3、tableView动态布局UITableViewAutomaticDimension

iOS 8 之后可以使用的动态约束UITableViewAutomaticDimension,动态布局主要还是在cell 上控件的约束,使用方法:

    _messageTableView.rowHeight = UITableViewAutomaticDimension;
    [_messageTableView setEstimatedRowHeight:44];

或者

- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {
    return UITableViewAutomaticDimension;
}

自定义Cell的时候,将约束写在- (void)layoutSubviews可能会存在一个问题那就是约束未被初始化导致第一次加载tableView的时候高度不准确,在拖动或者滚动后动态高度才是正确的,看下面效果图:

未初始化约束效果图

造成这个的原因是:将约束写在- (void)layoutSubviews方法时需要另外调用[self layoutIfNeeded];来更新约束,但是写在- (void)layoutSubviews容易导致内存的增加,因为每次调用[self layoutIfNeeded];会增加mas_makeConstraints增加约束导致内存消耗,不是很建议该写法:

        [_titleLabel setText:@"这是系统消息"];
        [_contentLabel setText:@"摸摸摸爱妃玛科技发福了卡积分老卡机啊福利肯德基奥拉夫控件的啊发送的发送啊发送的发送发送的发送方公司的反感"];
        [_timeLabel setText:@"2017-09-08 10:10"];
        // 约束如果是写在layoutSubviews别漏了这句代码,调用约束。
        // [self layoutIfNeeded];

建议写法:
将cell 的约束写在只加载一次的方法中例如- (instancetype)initWithStyle:(UITableViewCellStyle)style reuseIdentifier:(NSString *)reuseIdentifier

4、更新约束

很多时候还是需要根据数据来对布局进行处理,从而需要对约束进行更新。
原本的约束代码:

       [_fullView mas_makeConstraints:^(MASConstraintMaker *make) {
            make.top.equalTo(_couponLabel.mas_bottom).mas_offset(SCR_W * 10);
            make.left.equalTo(_iconView.mas_right).offset(SCR_W * 10);
            make.size.mas_equalTo(CGSizeMake(SCR_W * 25, SCR_W * 15));
        }];

在需要更新约束的地方,调用如下代码:

        //更新控件头部的位置
        [_fullView mas_updateConstraints:^(MASConstraintMaker *make) {
            make.top.equalTo(_dashedView.mas_bottom).mas_offset(SCR_W * 10);
        }];
        // 如果约束写在- (void)layoutSubviews, 需要立即更新请调用以下方法
        // [self layoutIfNeeded];

注意:更新之前跟更新之后的约束是要针对同个对象才能生效,例如需要更新make.left.equalTo(self.view.mas_left).mas_offset(10),更新后为make.left.equalTo(self.view.mas_left).mas_offset(50),而不能是make.left.equalTo(self.view.mas_right).mas_offset(10)

Masonry使用优化

缘由

使用Masonry已经有一段时间了,但是发现项目中tableView做了复用但是每次滚动,内存都会一直增加,对就是增加根本停不下来。开始怀疑人生了,毕竟复用就那些注意点,但是检查了十几遍的代码还是没发现问题。直到重复滚了好好几十遍的tableView后确认了一个问题:在更新cell上面的某个控件的时候会造成一个问题就是卡顿、内存增加。有了目标开始针对性查找,最后确认了一个事 —— 官方文档对updateConstraints的解释:
Custom views that set up constraints themselves should do so by overriding this method.When your custom view notes that a change has been made to the view that invalidates one of its constraints, it should immediately remove that constraint, and then call setNeedsUpdateConstraints to note that constraints need to be updated. Before layout is performed, your implementation of updateConstraints will be invoked, allowing you to verify that all necessary constraints for your content are in place at a time when your custom view’s properties are not changing.
重点是说在对没用的或失效的约束应该立即删除。而Masonry的mas_makeConstraints方法是添加约束。每添加一层,内存增长一次,你不删除,它就一直在。

解决

在需要重新增加整个控件的约束时使用mas_remakeConstraints而不是再次使用mas_makeConstraints,因为mas_remakeConstraints会先移除原本的约束,再重新添加。

对于只需要更新部分约束布局时则使用mas_updateConstraints

强调

mas_makeConstraints只会新增约束,不管原本是否已经有约束。
mas_remakeConstraints会先移除约束,再新增约束。

再尝试滚动tableView的性能得以优化,不会再增加内存。

技巧

1、子视图相对于父视图居中,可以考虑设置它的X、Y轴约束
2、子视图填满父视图,可以考虑设置子视图的四个边距为0
3、动态高度,可以只设置该视图的宽,高度由约束自动生成(动态高度,然后需要设置最大高度,好像是没法设置的)
就简单说这几点吧,具体的还是靠实践。多撸代码,也就熟能生巧了。
4、动画或者改变位置,可以将需要更改的某条约束定义为全局,然后根据需求进行更改。例如: _leftConstraint

    [_titleLabel mas_makeConstraints:^(MASConstraintMaker *make) {
       _leftConstraint = make.left.mas_equalTo(SCR_W(15));
        make.top.equalTo(@15);
        make.size.mas_equalTo(CGSizeMake(SCR_W(345), SCR_W(15)));
    }];

结语

总的介绍就这些吧(后续会更新),具体的还是需要看你自己多去尝试、实践才能孰能生巧。代码上可以简便一些,通用的布局,熟了可能一句约束就OK不熟的话可能需要拆分写好几句,才能达到相应的效果。

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

推荐阅读更多精彩内容

  • 一、前言 关于苹果的布局一直是我比较纠结的问题,是写代码来控制布局,还是使用storyboard来控制布局呢?以前...
    iplaycodex阅读 2,444评论 0 1
  • Masonry是一个轻量级的布局框架,拥有自己的描述语法,采用更优雅的链式语法封装自动布局,简洁明了并具有高可读性...
    3dcc6cf93bb5阅读 1,759评论 0 1
  • (一)Masonry介绍 Masonry是一个轻量级的布局框架 拥有自己的描述语法 采用更优雅的链式语法封装自动布...
    木易林1阅读 2,325评论 0 3
  • [置顶]iOS - Masonry使用中的一些整理 标签:iOS资源大全iOS常用方法iOS学习资料Masonry...
    DreamMakerSky阅读 3,161评论 0 4
  • 我所看到的听到的,有大部分人在自己的工作岗位上都是一个字“混”,年纪越大越这样。他们都没找到自己工作岗位上的乐趣。...
    今墨阅读 786评论 0 5