03 | Auto Layout 是怎么进行自动布局的,性能如何?
03 | Auto Layout 是怎么进行自动布局的,性能如何?
讲述:冯永吉
时长11:34大小10.57M
Auto Layout 的来历
Auto Layout 的生命周期
Auto Layout 性能问题
Auto Layout 的易用性
小结
课后小作业
赞 10
提建议
精选留言(107)
- 一剑寒潇置顶2019-03-16使用 Auto Layout 一定要注意多使用 Compression Resistance Priority 和 Hugging Priority,利用优先级的设置,让布局更加灵活,代码更少,更易于维护。 能否举一些详细的代码事例!!!
作者回复: 有的。你可以看我的这个 demo https://github.com/ming1016/ShowAutoLayout
共 3 条评论26 - 开发小能手2019-03-17关于 UI 的布局,是使用 frame 还是 Auto Layout ?这个问题困扰了我很久,在之前的团队中是建议使用 frame 的,原因是在 layoutSubviews 中做布局,将所有的子视图按照 UI 显示从上至下、从左至右的顺序来布局,感觉也很规范,代码也不会导致不易维护的程度。 我一直有这么一条原则:简单的 UI 使用 Auto Layout ,复杂的 UI 使用 frame。原因如下: 1、从代码量上来看,两种布局方式相差不大。有时候发现复杂的 UI 使用 Auto Layout 的话,代码量反而会变多,因为复杂的 UI 往往会有复杂的逻辑,比如根据数据的不同,部分 UI 的显示会有变动(比如某个子视图隐藏与显示, 会影响到其它视图的布局)。 2、会将那种仅做了一次约束之后,就可以不用做太多 update 的那种视为简单的布局,这种情况下使用 Auto Layout 还是挺方便的。 3、会将 Cell 的高度会随着数据的不同而不同的布局视为复杂的 UI 布局,这种情况下使用 Auto Layout 来布局,感觉就不合适。因为不管是 frame 还是 Auto Layout,都需要去计算高度,其实在计算 高度的时候,所有的子视图的 frame 都已经决定了,这种情况下,直接使用 frame 会比较精简。 4、我见过这样的代码:动态的通过文本,计算出尺寸之后,再使用 Auto Layout 进行 update,感觉太没有这个必要了。 以上是我对 UI 布局的一点思考,请教大神:凭您的经验,这样的思考有什么不合理之处么?(简单的 UI 使用 Auto Layout ,复杂的 UI 使用 frame。)展开
作者回复: 这样想没有问题
共 4 条评论58 - 徐秀滨2019-03-16针对Auto Layout的生命周期,我是这么理解的: Auto Layout拥有一套Layout Engine引擎,由它来主导页面的布局。App启动后,主线程的Run Loop会一直处于监听状态,当约束发生变化后会触发Deffered Layout Pass(延迟布局传递),在里面做容错处理(约束丢失等情况)并把view标识为dirty状态,然后Run Loop再次进入监听阶段。当下一次刷新屏幕动作来临(或者是调用layoutIfNeeded)时,Layout Engine 会从上到下调用 layoutSubviews() ,通过 Cassowary算法计算各个子视图的位置,算出来后将子视图的frame从Layout Engine拷贝出来,接下来的过程就跟手写frame是一样的了。 这样可对?展开
作者回复: 对
共 2 条评论40 - burning😕🙄2019-03-16没有讲到 cell 里面 label的高度自适应问题,这个在layout应该使用频率比较高吧
作者回复: 具体使用,可以看我这篇文章,里面涉及到 https://github.com/ming1016/study/wiki/Masonry 我把相关内容截取到这里 主要是UILabel的高度会有变化,所以这里主要是说说label变化时如何处理,设置UILabel的时候注意要设置preferredMaxLayoutWidth这个宽度,还有ContentHuggingPriority为UILayoutPriorityRequried CGFloat maxWidth = [UIScreen mainScreen].bounds.size.width - 10 * 2; textLabel = [UILabel new]; textLabel.numberOfLines = 0; textLabel.preferredMaxLayoutWidth = maxWidth; [self.contentView addSubview:textLabel]; [textLabel mas_makeConstraints:^(MASConstraintMaker *make) { make.top.equalTo(statusView.mas_bottom).with.offset(10); make.left.equalTo(self.contentView).with.offset(10); make.right.equalTo(self.contentView).with.offset(-10); make.bottom.equalTo(self.contentView).with.offset(-10); }]; [_contentLabel setContentHuggingPriority:UILayoutPriorityRequired forAxis:UILayoutConstraintAxisVertical]; 如果版本支持最低版本为iOS 8以上的话可以直接利用UITableViewAutomaticDimension在tableview的heightForRowAtIndexPath直接返回即可。 tableView.rowHeight = UITableViewAutomaticDimension; tableView.estimatedRowHeight = 80; //减少第一次计算量,iOS7后支持 - (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath { // 只用返回这个! return UITableViewAutomaticDimension; } 但如果需要兼容iOS 8之前版本的话,就要回到老路子上了,主要是用systemLayoutSizeFittingSize来取高。步骤是先在数据model中添加一个height的属性用来缓存高,然后在table view的heightForRowAtIndexPath代理里static一个只初始化一次的Cell实例,然后根据model内容填充数据,最后根据cell的contentView的systemLayoutSizeFittingSize的方法获取到cell的高。具体代码如下 //在model中添加属性缓存高度 @interface DataModel : NSObject @property (copy, nonatomic) NSString *text; @property (assign, nonatomic) CGFloat cellHeight; //缓存高度 @end - (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath { static CustomCell *cell; //只初始化一次cell static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ cell = [tableView dequeueReusableCellWithIdentifier:NSStringFromClass([CustomCell class])]; 还有这篇 https://ming1016.github.io/2015/11/03/deeply-analyse-autolayout/ 会讲的更详细些
29 - Sean Ma2019-03-16我想了解下自动布局会在哪一个时机下能够拿到相应的frame和center,因为有时候做完约束,我想在viewdidlayer前拿到其frame。另外自动布局怎么做动画
作者回复: 文章有提到。Layout Engine 会从上到下调用 layoutSu... 这个部份。 动画的话。因为布局约束就是要脱离frame这种表达方式的,可是动画是需要根据这个来执行,这里面就会有些矛盾,不过根据前面说到的布局约束的原理,在某个时刻约束也是会被还原成frame使视图显示,这个时刻可以通过layoutIfNeeded这个方法来进行控制。具体代码如下 [aniView mas_makeConstraints:^(MASConstraintMaker *make) { make.top.bottom.left.right.equalTo(self.view).offset(10); }]; [aniView mas_updateConstraints:^(MASConstraintMaker *make) { make.top.equalTo(self.view).offset(30); }]; [UIView animateWithDuration:3 animations:^{ [self.view layoutIfNeeded]; }];
15 - power2019-03-18不知道大家对视图依赖的处理有没有很好的方法:A依赖B,B依赖C,C依赖D,BC有可能不存在,所以D的依赖有可能是ABC,这种有没有好的布局思路呢共 6 条评论12
- 徐扬2019-03-18autolayout 的布局代码应该写在什么地方,是在view 的layoutsubview 方法里面写,还是在view 加载完成后写,那个更合理呢?
作者回复: 推荐用 Masonry 和 SnapKit
共 3 条评论11 - Hello灬麦德姆2019-03-16有一会面试面试官问我AutoLayout怎么实现的,我懵逼了,相见恨晚啊!共 3 条评论11
- drunkenMouse2019-03-19把循环这部分总结了一下: 触发约束变化 —> Layout Engine就需要重新计算布局,会先获取到当前的布局,调用SuperView.SetNeedLayout() —> Deffered Layout Pass进行监听 —> Layout Engine 从上到下调用LayoutSubViews(),通过Cassowary算法计算各个子视图的位置,算出来后将子视图的Frame从Layout Engine里拷贝出来 —> 和手写布局的绘制、渲染一样。8
- xilie2019-03-18Hi,看了你写的“AutoLayout情况如何计算UITableView的变高高度”,问个问题: 有时候cell里面的一些 label 会在不同情况下切换显示和隐藏,这时 label 的上下间距,会出现异常,比如: 竖着排列3个 label: a-b-c,间距都是20,这时 b 隐藏,c 和 a 的间距是20,设计要求是10,这种怎么处理?展开
作者回复: layoutsubview 里可以得到 frame 然后判断进行hidden处理
共 3 条评论5 - geeklyc2019-03-17看了篇文章比较几种不同计算cell高度性能的问题,从好到坏:手动frame+计算高度,AutoLayout+计算高度,AutoLayout+自动计算。铭神,你的建议是?现在三种情况都用了下
作者回复: cell 情况复杂时用手动frame+计算高度,简单的用AutoLayout+自动计算
共 3 条评论4 - wlgemini2019-03-18我也写了一个基于Auto layout的swift框架,在这里:https://github.com/wlgemini/Driftwood
作者回复: 棒棒的
3 - CoderWGB2019-03-16Flexbox布局iOS开发用的少 一般都是frame或者Masonry3
- 吃蘑菇的大灰狼2019-03-16UIStackView的推出从iOS9到现在已经很久了, 他确实更方便布局,但是在平常的开发里面,貌似没看到有很多小伙伴在用, 记得有一篇文章提到,苹果的开发人员建议 ,如果能用UIStackView的,尽量都用UIStackView. 之前不知道原因,读了这篇戴老师的文章才知道UIStackView布局的原理。期待后续的文章。3
- 轩2019-03-27关于 Autolayout,iOS 12 具体做了哪些优化呢?
作者回复: 参见 WWDC 2018 202 Session
2 - Bo.zhang2019-03-24戴神 为什么https://github.com/ming1016/STMAssembleView 下载后程序运行后出link错误 怎么修改呢?共 1 条评论2
- Geek_45fc022019-03-23Autolayout+XIB或Storyboard对开发效率的提升是巨大的,真正的所见即所得,甚至对于独立开发而言很多场景都可以直接把storyboard当作画板来使用,省去了在sketch上画UI的时间。如果iOS没有XIB或者storyboard,Flutter可能会比现在更受欢迎。2
- 张松超2019-03-17不推荐使用xib吗,还是不推荐使用拖线的方式
作者回复: 简单页面用 xib 没什么问题
2 - 徐秀滨2019-03-16请问,使用xib画页面能够快速提高开发效率,但是怎么在MVC架构上合理使用呢?比如新建一个VC,勾上xib选项后,那么就会创建一个xib文件,这时候xib文件与VC位于同一目录下。 1. 从文件分类来讲,xib属于View层,VC属于C层,这就不对了。 2. 而且xib上有响应动作的话该怎么处理才能做到符合MVC分离,且不失了直接拉线写method这种便利性呢?展开
作者回复: VC.view 了解下。以外的才是 C 层
2 - Anyeler2019-03-16主要公司的APP支持iOS8.0以上,所以UIStackView比较少用
作者回复: 孙源的 FDStackView 了解下。 https://github.com/forkingdog/FDStackView
2