2

我有一个在 UIStoryboard 中使用 UITableView 设置的视图控制器。UITableView 在所有四个边上都被限制在控制器的 SafeArea 中。

这个问题好像以前也有人问过,但是每次我发现这个问题,都是因为viewDidLayoutSubviews发生了变化,导致了循环的发生。我有 viewDidLayoutSubviews 和基本 UITableViewCells 的空白实现(无笔尖)。然而在 iOS 11、Xcode 9.2 中,向上轻弹 tableview 会导致 viewDidLayoutSubviews 在无限循环中被调用。这是我的实现:

- (void)viewDidLayoutSubviews {

    NSLog(@"viewDidLayoutSubviews");
    [super viewDidLayoutSubviews];
}

我一生都无法弄清楚这是怎么发生的。

编辑:控制器有一个 NavigationController 并且是 UISplitViewController 的主控制器,它嵌入在 BannerViewController 中(来自 iAdSuite 示例)。

编辑:这发生在模拟器上所有设备的 iOS 11(直到 11.2)上。不会发生 iOS 10.3。viewDidLayoutSubviews在 iOS 10.3 中甚至没有调用。当我简单地使用导航控制器和 UIViewController/TableView 代替 UISplitViewController/BannerViewController 时,我得到了相同的结果。所以似乎这个问题不是在那里介绍的,而是来自 iOS 11 中正在发生的事情。

编辑:所以这很奇怪 - 似乎只有在为导航控制器选择大标题时才会发生!

if (@available(iOS 11.0, *)) {
    self.navigationController.navigationBar.prefersLargeTitles = YES;
    self.navigationItem.largeTitleDisplayMode = UINavigationItemLargeTitleDisplayModeAlways;
}

编辑:这个问题很容易重现。我下载了 Apple 的 iAd Suite 示例代码,并使用了 SplitNavigationController 项目。我更新了 iOS 11 的项目,包括更新故事板以使用安全区域布局指南。我将 MasterViewController 从 UITableViewController 更改为带有 UITableView 的 UIViewController,我将其限制为情节提要中的安全区域布局指南。我在 MasterViewController 中添加了 prefersLargeTitles 和 largeTitleDisplayMode。我在 viewDidLayoutSubviews 方法中添加了一个 NSLog 语句,因此可以很容易地观察到这种行为。我认为这是 iOS 11 的一个真正问题,我将尝试深入了解它,如果其他人可以提供任何想法或建议,我也将不胜感激。

编辑:matt 能够重现此问题并将其发布在 GitHub 项目中,可在以下链接中找到(感谢 matt!):InfiniteLayoutSubviewsBug

4

1 回答 1

3

编辑这绝对是 iOS 11 到至少 iOS 11.2 中的一个错误,但在 iOS 12 中,该错误不再存在。

看起来你发现了一个错误。我能够在一个最小的示例项目中重现该问题,该项目已在 GitHub 上发布:

https://github.com/mattneub/InfiniteLayoutSubviewsBug

下载项目并运行它。您将看到一个包含三行的简单表格。向上滚动表格视图并松开。观察控制台。我们viewDidLayoutSubviews每 1/60 秒一次,永远不断地重复调用 。

请注意,这不是由我们代码中的递归布局循环引起的。我的例子甚至没有调用super. 它所做的只是记录。这是运行时本身viewDidLayoutSubviews永远重复调用。发生这种情况时,我们的代码(除了print)不会运行。

其他观察:

  • 正如您正确观察到的,如果您更改示例以使导航栏不使用大标题,问题就会消失。

  • 如果您更改示例以便不使用自动布局定位表视图,问题就会消失。

  • 如果您更改示例(更详细地)以使视图控制器成为 UITableViewController 子类,那么问题就消失了;只是在滚动时,我们会收到一些重复的调用,viewDidLayoutSubviews但不是永远。

于 2017-12-21T04:47:13.547 回答