鍍金池/ 問答/C  iOS/ 為什么使用masonry不能立即獲取frame?

為什么使用masonry不能立即獲取frame?

第一段代碼:

[self.scrollView mas_makeConstraints:^(MASConstraintMaker *make) {
    make.edges.mas_equalTo(UIEdgeInsetsMake(0, 0, 0, 0));
}];
NSLog(@"%@", self.scrollView);

結(jié)果是:<UIScrollView: 0x7fcae504ba00; frame = (0 0; 0 0);

第二段代碼:

[self.scrollView mas_makeConstraints:^(MASConstraintMaker *make) {
    make.edges.mas_equalTo(UIEdgeInsetsMake(0, 0, 0, 0));
}];
// 延遲0.1秒后獲取frame
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.1 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
    NSLog(@"%@", self.scrollView);
});

結(jié)果是:<UIScrollView: 0x7fad83820a00; frame = (0 0; 375 667);

問:

為什么延遲0.1秒后就能獲取到正確的frame了?

回答
編輯回答
別逞強

autolayout的話,如果你是在一個viewController里面,一般是在viewDidLayoutSubviews才能生效,如果是一個view的話,一般則在layoutSubviews里面才生效

2018年8月20日 03:40
編輯回答
喜歡你

他是用的AutoLayout 啊,自動布局 是拿不到精確的frame 的吧

2018年6月5日 15:49
編輯回答
墨沫

先來一句概括的話,咱們寫代碼也是要講究基本法的. 嗯,應(yīng)該是講究基本流程.

首先需要明白從 Constraints 對 最后的布局完成,肯定有一個轉(zhuǎn)化成 frame 的過程.
上面一些回答中說到的 viewLayout 即是這個過程已經(jīng)完成時調(diào)用相應(yīng)方法的地方.

可能你還有疑問就是,所有的 UI 代碼都是在主線程工作的.
那么當調(diào)用 NSLog 的時候是不是隨著 makeConstraints 或 updateConstraint 的調(diào)用就完成了呢?
因為我們知道隨著約束的變化, autoLayoutEngine 肯定會因此來更新對應(yīng)的約束的值的. 如果這一切是同步的.
那在你調(diào)用 NSLog 時 它的 frame 就應(yīng)該變化了.

但是并不是的, UI 系統(tǒng)并不是隨著每一個 UI 變化的調(diào)用同步執(zhí)行的.
UI刷新是有固定的頻率的. 假設(shè)以 30 FPS 為例. 也就是系統(tǒng)可能需要過 0.03秒之后才會刷新 UI,
處理你對 UI 的請求變化.
你提到的 0.1 秒其實已經(jīng)過了好幾個 VSYNC階段了,所以 frame 已經(jīng)計算出來了.
其實你過 0.05秒也是可以得到其 frame 的.

核心點在于 UI刷新的工作流程, VSYNC

2017年5月11日 18:53
編輯回答
愿如初

后來我自己總結(jié)了一篇簡書:http://www.jianshu.com/p/e71b...

另:我在S.O.上問了一下,有人給出的答案-
https://stackoverflow.com/que...

2018年5月29日 14:15
編輯回答
不將就

[self.scrollView mas_makeConstraints:^(MASConstraintMaker *make){
    make.edges.mas_equalTo(UIEdgeInsetsMake(0, 0, 0, 0));
}];

下面加上這句就可以了,

[self.scrollView.superview layoutIfNeeded];

這樣就能獲取到self.scrollViewframe了。

2017年8月3日 01:22
編輯回答
莫小染

推薦一篇 博文 ,講得挺不錯的!

2017年6月7日 09:45
編輯回答
念舊

等到調(diào)用了viewWillLayoutSubviews 才會拿到你想要的frame,需要時間

2018年5月26日 18:49
編輯回答
蟲児飛

結(jié)合Masonry源碼,在View+MASAdditions.m文件中的- (NSArray *)mas_makeConstraints:(void(^)(MASConstraintMaker *))block方法里打個斷點,就很清晰了。
第一段代碼,先走

[self.scrollView mas_makeConstraints:^(MASConstraintMaker *make) {
    make.edges.mas_equalTo(UIEdgeInsetsMake(0, 0, 0, 0));
}];

mas_makeConstraints:方法中注冊block,調(diào)用MASConstraintMaker的 install方法,建立約束。
然后走

NSLog(@"%@", self.scrollView);

再到mas_makeConstraints方法中實現(xiàn)block。
因為Masonry中block注冊的時間是作用域結(jié)束,就是大括號末尾。
dispatch_after方法中,block注冊的時機,要更延后一個設(shè)定的間隔。
為什么Masonry中block注冊的時間是這樣的,請看網(wǎng)上的Masonry源碼解析。
通過斷點,能夠很直觀地看到。

2017年2月11日 00:33