Xcode 10.3 (10G8) 使用编译模式 = "Whole Module" 破坏应用程序
Posted
技术标签:
【中文标题】Xcode 10.3 (10G8) 使用编译模式 = "Whole Module" 破坏应用程序【英文标题】:Xcode 10.3 (10G8) breaks app with compilation mode = "Whole Module" 【发布时间】:2019-08-09 19:43:31 【问题描述】:我将我的 ios 应用程序的 7.6 版推送到 App Store 并注意到该应用程序突然包含许多在调试过程中没有出现的错误(奇怪的行为)。
由于该应用在 xCode 9.x 中完美运行,我强烈怀疑这些问题是从 Xcode 10.3 开始出现的。
我正在使用 Swift 4.2(Swift 5 的转换待下次更新)
经过几个小时的调查,我找到了问题:当编译模式 =“whole module
”时出现错误,而当设置为“incremental
”时,错误消失。在调试模式下(当应用程序用完 Xcode 时),编译模式设置为“incremental
”以释放其“whole module
”(这是我怀疑在 Xcode 10.x 中创建新项目时的标准配置) 这解释了为什么我们在调试测试期间没有发现问题。
另请注意,更改为旧版构建系统并不能解决问题。只有设置 Compilation mode = "incremental
" 才能解决问题。
分析:
我跟踪问题是因为我的所有 TableView 都没有调用委托。我有以下简单的层次结构:
ViewTableRoot 的代码:
class ViewTableRoot : UITableView, UITableViewDelegate, UITableViewDataSource
var didScrollToOffset : ( (CGFloat) -> Void )?
var didEndScrolling : ( (CGFloat) -> Void )?
var didChangeEditing : ( ( ) -> Void )?
//MARK: lifecycle
required init?(coder aDecoder: NSCoder)
super.init(coder: aDecoder)
setup()
override init(frame: CGRect, style: UITableView.Style)
super.init(frame: frame, style: style)
setup();
func setup()
//set the corner radius of the layer so that the sliding of the cells underneath the rounded headers does not show up
layer.cornerRadius = 5
//setup myself as delegate and data source
delegate = self
dataSource = self
deinit
let className = String(describing: self)
log.debug("**********\(className)")
//MARK: - public API
override func setEditing(_ editing: Bool, animated: Bool)
super.setEditing(editing, animated: animated)
didChangeEditing?()
//MARK: - scrollview delegate
func scrollViewDidScroll(_ scrollView: UIScrollView)
//because we are also getting events when swiping on the cells, we need to see the difference between
//swipig on the cell and swiping in the "actual" table => we do this by checking the frame size
guard scrollView.frame == frame else return
didScrollToOffset?(scrollView.contentOffset.y)
func scrollViewDidEndDragging(_ scrollView: UIScrollView, willDecelerate decelerate: Bool)
//because we are also getting events when swiping on the cells, we need to see the difference between
//swipig on the cell and swiping in the "actual" table => we do this by checking the frame size
guard scrollView.frame == frame else return
if !decelerate
didEndScrolling?(scrollView.contentOffset.y)
func scrollViewDidEndDecelerating(_ scrollView: UIScrollView)
//because we are also getting events when swiping on the cells, we need to see the difference between
//swipig on the cell and swiping in the "actual" table => we do this by checking the frame size
guard scrollView.frame == frame else return
didEndScrolling?(contentOffset.y)
//MARK: - UITableViewDataSource
func numberOfSections(in tableView: UITableView) -> Int
preconditionFailure("Must be implemented by derrived class")
func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int
preconditionFailure("Must be implemented by derrived class")
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell
preconditionFailure("Must be implemented by derrived class")
//MARK: - UITableViewDelegate
func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath)
ViewTableSettings 的代码:
class ViewTableSettings : ViewTableRoot
var settings : [[Setting]]?
didSet
reloadData()
var didPressSetting : ((Setting, CGRect) -> (Void))?
//MARK: lifecycle
override func setup()
super.setup()
log.debug("delegate : \(delegate)")
//register xibs
register(CellTableSetting.nib, forCellReuseIdentifier: CellTableSetting.reuseIdentifier)
//MARK: - UITableView
override func numberOfSections(in tableView: UITableView) -> Int
let count = settings?.count ?? 0
log.debug("count: \(count)")
log.debug("delegate : \(delegate)")
return count
override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int
let count = settings?[section].count ?? 0
log.debug("count: \(count)")
log.debug("delegate : \(delegate)")
return count
override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell
log.debug("delegate : \(delegate)")
//ask for a new cell
let cell = tableView.dequeueReusableCell(withIdentifier: CellTableSetting.reuseIdentifier, for: indexPath) as! CellTableSetting
guard let setting = settings?[indexPath.section][indexPath.row] else
preconditionFailure("Asking CellTableSetting but no Setting model defined")
//load up!
cell.setting = setting
cell.lastCell = indexPath.section != numberOfSections - 1 ? false : indexPath.row == (numberOfRows(inSection:indexPath.section) - 1)
//return cell to use
return cell
func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat
log.debug("-")
return CellTableSetting.height
override func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath)
log.debug("-")
tableView.deselectRow(at:indexPath, animated: true)
guard let setting = settings?[indexPath.section][indexPath.row] else
return
didPressSetting?(setting,rectForRow(at: indexPath))
func tableView(_: UITableView, viewForFooterInSection: Int) -> UIView?
log.debug("-")
guard viewForFooterInSection < (numberOfSections-1) else
let version = UILabel()
version.text = UIApplication.AppVersionAndBuildPrettyPrint
version.font = UIFont.defaultBoldFont(size: 12)
version.textColor = PaintCode.mainLightGray_a50
version.textAlignment = .center
return version
let v = UIView()
v.backgroundColor = PaintCode.mainLightGray_a50
return v
func tableView(_: UITableView, heightForFooterInSection: Int) -> CGFloat
log.debug("-")
return heightForFooterInSection < (numberOfSections-1) ? 5 : 40
如您所见,ViewTableRoot
声明符合UITableViewDelegate
(也符合UITableViewDataSource
,但目前还不是问题)
委托实际上在ViewTableRoot
中分配给self
,但实际的委托功能在派生的ViewTableSettings
中实现(同样,这在Xcode 9.x 中完美运行)
当编译模式=“Whole Module
”时,这些委托函数没有被调用=>这是错误
当设置为“incremental
”时,这些委托函数可以正常调用!
为了更深入地了解该问题,我进行了其他测试:
切换到“旧版构建系统”(通过 Xcode/file/project 设置)不能解决问题;只要启用Whole Module
,问题仍然存在
当我在 ViewTableRoot 中创建空委托函数并在 ViewTableSettings 中覆盖它们时,它确实有效:-o
我确实在 ViewTableSettings
中验证了委托确实设置为 ViewTableSettings
的实例而不是 ViewTableRoot(在这种情况下不会实现任何委托函数)
我的想法
我觉得我偶然发现了(新的?)构建系统中的一个错误? 还有其他人遇到过类似问题吗?【问题讨论】:
【参考方案1】:是的,我和你有同样的问题。当编译模式为“Whole Module”时,Collectionview 控制器会混乱,但 Tableview 控制器不会。 result on simulator device。也许你可以尝试直接使用 UITableViewController 而不是遵循 TableView 协议。
【讨论】:
【参考方案2】:安装了最新的 Xcode 11.4 (11E146) 和这个版本中要解决的问题接缝。我重新启用了整个模块优化,并且所有接缝都按预期工作。所以...原来是 XCdoe 的一个 bug!
【讨论】:
以上是关于Xcode 10.3 (10G8) 使用编译模式 = "Whole Module" 破坏应用程序的主要内容,如果未能解决你的问题,请参考以下文章
“if (@available(iOS 13.0, *))” 在 Xcode 10.3 中无法编译