很多人习惯重写UIView的layoutSubview方法,在里面写View的布局代码,期望视图在数据发生变化,或系统主题改变时,页面的布局能够执行同样的代码逻辑,保证最终展示结果符合预期。
- (void)layoutSubviews{
[super layoutSubviews];
CGSize size = self.bounds.size;
CGSize longTitleSize = [self sizeThatFits:size title:[self _longLogInTitle]];
NSString *title = (longTitleSize.width <= size.width ?[self _longLogInTitle:[self _shortLogInTitle]);
if (![title isEqualToString:[self titleForState:UIControlStateNormal]]) {
[self setTitle:title forState:UIControlStateNormal];
}
}
那么一个简单问题就来了,layoutSubView方法究竟是什么情况下会触发调用?
触发layoutSubView被调用的时机总结如下:
1. UIView frame发生变化
2. 调用UIView的addSubview方法
3. 主动调用UIView的setNeedLayout方法
4. 旋转屏幕
5. UIScrollView滚动使得view需要重绘
需要强调的是,layoutSubView是由iOS底层异步回调的,用户代码中调用setFrame、addSubview、setNeedLayout等方法,并不会导致layoutSubView方法在当前消息循环中被立刻调用,对layoutSubView方法的调用是会延迟到下一个或更靠后的消息循环中去。
我们知道, iOS页面渲染上屏主要包含Layout->Display->Prepare->Commit四个步骤,而屏幕的渲染是由硬件驱动的。大部分iOS设备的vSync信号产生的周期是16ms一次,这是layoutSubView会被异步调用的底层原因。