使用Autolayout时在layoutSubviews中添加子视图是不是正确
Posted
技术标签:
【中文标题】使用Autolayout时在layoutSubviews中添加子视图是不是正确【英文标题】:Is it right to add subview in layoutSubviews when using Autolayout使用Autolayout时在layoutSubviews中添加子视图是否正确 【发布时间】:2017-02-20 03:11:32 【问题描述】:我写了这样的代码,效果很好
@interface SubView()
@property(nonatomic,strong) UIButton* btn;
@end
@implementation SubView
- (void)layoutSubviews
[super layoutSubviews];
[self.btn removeFromSuperview];
self.btn = [UIButton buttonWithType:UIButtonTypeCustom];
self.btn.backgroundColor = [UIColor greenColor];
[self.btn setTitle:@"btn" forState:UIControlStateNormal];
self.btn.translatesAutoresizingMaskIntoConstraints = NO;
[self addSubview:self.btn];
NSLayoutConstraint* centerXConstraint = [self.btn.centerXAnchor constraintEqualToAnchor:self.centerXAnchor];
NSLayoutConstraint* centerYConstraint = [self.btn.centerYAnchor constraintEqualToAnchor:self.centerYAnchor];
[self addConstraint:centerXConstraint];
[self addConstraint:centerYConstraint];
@end
但在我看来,在使用Autolayout时,系统会分两步做布局1 Update Pass 2 Layout Pass,layoutSubviews在step 2(Layout Pass)中
所以如果我们在 layoutSubviews 中添加子视图,似乎它会改变视图的约束,所以我们需要再次进行更新传递和布局传递,所以它会生成一个无限循环..
但事实上,这段代码运行良好,那我哪里错了?
【问题讨论】:
添加subView
不会改变视图的约束,因此layoutSubviews
不会被调用。但是layoutSubviews
不是添加子视图的好地方,最好在初始化器中添加。
你的意思是上面的 centerXConstraint 不是视图的约束,而是子视图的约束?所以视图的 layoutSubviews 可以正确布局
是的,它只是将按钮与视图的 centerX 对齐,这不会影响视图本身的位置/大小。 layouSubiviews
会在视图的框架更改时调用,或者如果您使用 setNeedsLayout
或 layoutIfNeeded
手动触发布局。
好的,知道了,非常感谢@EricQian
【参考方案1】:
请记住,layoutSubviews
将被多次调用。每次删除按钮并重新添加它有什么意义?
另外,layoutSubviews
是自动布局系统遵循约束的地方。那么在layoutSubviews
中设置按钮的约束有什么意义呢?
即使它似乎在工作,你所做的一切在layoutSubviews
中都没有任何意义。执行此任务一次,然后在其他地方执行。
【讨论】:
感谢@matt,我在 updateConstraints 中看到了一些代码(就像我在上面发布的那样),所以在 updateConstraints 中添加/删除 subView 也是一个坏习惯? @ximmyxiao 是的。在其他地方做,比如在 viewDidLoad 中。 @Shebuka 如果它在自定义 uiview 中,似乎 init 方法将是做这些事情的唯一好地方? "所以在 updateConstraints 中添加/删除 subView 也是一个坏习惯" 这不是重点。关键是 your 代码删除了btn
视图,然后再次将其添加回来。那是愚蠢的部分。您只需要添加这个btn
视图一次,所以只需要添加一次就可以了。以上是关于使用Autolayout时在layoutSubviews中添加子视图是不是正确的主要内容,如果未能解决你的问题,请参考以下文章
使用AutoLayout布局适配时,如何提前获得AutoLayout完成适配后的子控件的真实frame
代码加约束VFL语法的详细使用介绍(代替Autolayout进行布局,比Autolayout更简单明了,生动直观)