ios - 自 iOS 9 以来 UITableView 的滚动性能不佳(之前工作得很好)

标签 ios swift uitableview autolayout uitextview

自从升级到 iOS 9 后,之前运行良好的代码开始在滚动时显示出非常糟糕的性能。

我试图从头开始创建一个最小的示例,省略了大部分单元的配置以及各种缓存和内容生成代码。然而,代码仍然不能快速阅读,并且分析显示不同:/

根据 Apple 开发者论坛(编辑:我发现了 link ),这可能是由于自动布局发生了变化,并且在它的 superView 后面有大量的布局 View 使它不那么糟糕,仍然不够好,无法发布。

我怀疑问题出在 UIKit 中,尤其是在自动布局和 UITextView 中,因为一旦我开始设置 cell,滚动就会立即变粗。 textView.text.

有没有人做过类似的观察?对于许多人来说,这似乎并不是一个问题。

有什么建议可以使这更顺利吗?使用 UILabels 修复它,但在现实生活中属性字符串中有链接是可点击的,所以我认为没有办法避免 UITextView

我正在使用自定义的 UITableViewCell 子类,我也对其进行了简化,但由于程序化的自动布局,它们仍然值得一读:

class AbstractRegularTimelineCell: UITableViewCell {
    weak var iconView : UIImageView!
    weak var headerLabel : UILabel!
    weak var timeLabel : UILabel!
    weak var regularContentContainer : UIView!
    weak var contentHeightConstraint : NSLayoutConstraint!
    required init?(coder aDecoder: NSCoder) {
        super.init(coder: aDecoder)
        commonInit()
    }

    override init(style: UITableViewCellStyle, reuseIdentifier: String?) {
        super.init(style: style, reuseIdentifier: reuseIdentifier)
        commonInit()
    }

    private func commonInit() {
        self.selectionStyle = .None
        // add the UI elements needed for all regular cells.

        let iv = UIImageView(/*image: UIImage(named: "icon")*/)
        iv.translatesAutoresizingMaskIntoConstraints = false
        self.contentView.addSubview(iv)
        self.iconView = iv
        iv.setContentCompressionResistancePriority(1000, forAxis: .Horizontal)
        iv.setContentCompressionResistancePriority(1000, forAxis: .Vertical)


        let hl = UILabel()
        hl.numberOfLines = 0
        hl.translatesAutoresizingMaskIntoConstraints = false
        self.contentView.addSubview(hl)
        self.headerLabel = hl
        hl.font = UIFont(name: "OpenSans-Semibold", size: UIFont.systemFontSize())

        let tiv = UIImageView(/*image: UIImage(named: "timestamp")*/)
        tiv.translatesAutoresizingMaskIntoConstraints = false
        self.contentView.addSubview(tiv)


        let tl = UILabel()
        tl.translatesAutoresizingMaskIntoConstraints = false
        self.contentView.addSubview(tl)
        self.timeLabel = tl

        tl.textColor = UIColor.lightGrayColor()
        tl.font = UIFont(name: "OpenSans", size: UIFont.systemFontSize()-3)

        let rcc = UIView()
        rcc.translatesAutoresizingMaskIntoConstraints = false
        self.contentView.addSubview(rcc)
        self.regularContentContainer = rcc
        rcc.setContentCompressionResistancePriority(1000, forAxis: .Vertical)
        rcc.setContentCompressionResistancePriority(100, forAxis: .Horizontal)
        self.contentView.sendSubviewToBack(rcc) // this might help, according to Apple Dev Forums




        // now, stitch the constraints together.
        let views = ["iv":iv, "hl":hl, "tl":tl, "rcc":rcc, "tiv":tiv]



        self.contentView.addConstraints(NSLayoutConstraint.constraintsWithVisualFormat("H:|-(15)-[iv]-(15)-[hl]-(>=15)-|", options: NSLayoutFormatOptions.DirectionLeadingToTrailing, metrics: [:], views: views))

        self.contentView.addConstraints(NSLayoutConstraint.constraintsWithVisualFormat("V:|-(15)-[iv]", options: NSLayoutFormatOptions.DirectionLeadingToTrailing, metrics: nil, views: views))

        self.contentView.addConstraints(NSLayoutConstraint.constraintsWithVisualFormat("V:|-(17)-[hl]-(8)-[rcc]", options: NSLayoutFormatOptions.DirectionLeadingToTrailing, metrics: nil, views: views))


        var otherConstraints = [NSLayoutConstraint]()
        var c = NSLayoutConstraint(item: tiv, attribute: .Top, relatedBy: .Equal, toItem: rcc, attribute: .Bottom, multiplier: 1, constant: 8)

        c.priority = 600
        otherConstraints.append(c)

        c = NSLayoutConstraint(item: tiv, attribute: .Top, relatedBy: .GreaterThanOrEqual, toItem: rcc, attribute: .Bottom, multiplier: 1, constant: 6)

        otherConstraints.append(c)

        c = NSLayoutConstraint(item: tl, attribute: .Leading, relatedBy: .Equal, toItem: tiv, attribute: .Trailing, multiplier: 1, constant: 6)

        otherConstraints.append(c)

        c = NSLayoutConstraint(item: tiv, attribute: .CenterY, relatedBy: .Equal, toItem: tl, attribute: .CenterY, multiplier: 1, constant: 1)

        otherConstraints.append(c)

        c = NSLayoutConstraint(item: self.contentView, attribute: .Bottom, relatedBy: .Equal, toItem: tiv, attribute: .Bottom, multiplier: 1, constant: 10)

        otherConstraints.append(c)

        c = NSLayoutConstraint(item: tiv, attribute: .Leading, relatedBy: .Equal, toItem: hl, attribute: .Leading, multiplier: 1, constant: 0)

        otherConstraints.append(c)


        c = NSLayoutConstraint(item: rcc, attribute: NSLayoutAttribute.Height, relatedBy: .LessThanOrEqual, toItem: nil, attribute: .NotAnAttribute, multiplier: 1, constant: 10000)

        otherConstraints.append(c)
        self.contentHeightConstraint = c

        c = NSLayoutConstraint(item: self.contentView, attribute: .Trailing, relatedBy: .GreaterThanOrEqual, toItem: rcc, attribute: .Trailing, multiplier: 1, constant: 15)
        otherConstraints.append(c)

        c = NSLayoutConstraint(item: rcc, attribute: .Leading, relatedBy: .Equal, toItem: hl, attribute: .Leading, multiplier: 1, constant: 0)
        otherConstraints.append(c)

        self.contentView.addConstraints(otherConstraints)
    }

}



class ExampleCell: AbstractRegularTimelineCell {
    weak var textView : UITextView!
    override required init(style: UITableViewCellStyle, reuseIdentifier: String?) {
        super.init(style: style, reuseIdentifier: reuseIdentifier)
        self.setupTextView()
    }

    required init?(coder aDecoder: NSCoder) {
        super.init(coder: aDecoder)
        self.setupTextView()
    }
    private func setupTextView() {
        let t = UITextView()
        t.translatesAutoresizingMaskIntoConstraints = false
        self.regularContentContainer.addSubview(t)
        self.regularContentContainer.addConstraints(NSLayoutConstraint.constraintsWithVisualFormat("H:|[t]|", options: NSLayoutFormatOptions.DirectionLeadingToTrailing, metrics: [:], views: ["t":t]))
        self.regularContentContainer.addConstraints(NSLayoutConstraint.constraintsWithVisualFormat("V:|[t]|", options: NSLayoutFormatOptions.DirectionLeadingToTrailing, metrics: [:], views: ["t":t]))
        self.textView = t
        t.scrollEnabled = false
    }
}

较短的 View Controller 代码:

class ViewController: UIViewController, UITableViewDataSource, UITableViewDelegate {
    weak var tableView : UITableView!
    override func viewDidLoad() {
        super.viewDidLoad()
        let t = UITableView()
        t.registerClass(ExampleCell.classForCoder(), forCellReuseIdentifier: "example")
        t.translatesAutoresizingMaskIntoConstraints = false
        self.view.addSubview(t)
        self.tableView = t
        self.view.addConstraints(NSLayoutConstraint.constraintsWithVisualFormat("H:|[t]|", options: NSLayoutFormatOptions.DirectionLeadingToTrailing, metrics: [:], views: ["t":t]))
        self.view.addConstraints(NSLayoutConstraint.constraintsWithVisualFormat("V:|[t]|", options: NSLayoutFormatOptions.DirectionLeadingToTrailing, metrics: [:], views: ["t":t]))
        t.delegate = self
        t.dataSource = self
        t.estimatedRowHeight = 350
    }


    func numberOfSectionsInTableView(tableView: UITableView) -> Int {
        return 1
    }
    func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return 300
    }
    func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
        let cell : ExampleCell = tableView.dequeueReusableCellWithIdentifier("example", forIndexPath: indexPath) as! ExampleCell
        cell.timeLabel.text = "probably never"
        switch indexPath.row % 3 {
        case 0:
            cell.headerLabel.text = "First paragraph (Part one of two)"
            cell.textView.text = "As any dedicated reader can clearly see, the Ideal of practical reason is a representation of, as far as I know, the things in themselves; as I have shown elsewhere, the phenomena should only be used as a canon for our understanding. These paralogisms of practical reason are what first give rise to the architectonic of practical reason."
        case 1:
            cell.headerLabel.text = "First paragraph, 2/2"
            cell.textView.text = "As will easily be shown in the next section, reason would thereby be made to contradict, in view of these considerations, the Ideal of practical reason, yet the manifold depends on the phenomena. Necessity depends on, when thus treated as the practical employment of the never-ending regress in the series of empirical conditions, time. Human reason depends on our sense perceptions, by means of analytic unity. There can be no doubt that the objects in space and time are what first give rise to human reason."
        default:
            cell.headerLabel.text = "Second paragraph"
            cell.textView.text = "Let us suppose that the noumena have nothing to do with necessity, since knowledge of the Categories is a posteriori. Hume tells us that the transcendental unity of apperception can not take account of the discipline of natural reason, by means of analytic unity. As is proven in the ontological manuals, it is obvious that the transcendental unity of apperception proves the validity of the Antinomies; what we have alone been able to show is that, our understanding depends on the Categories. It remains a mystery why the Ideal stands in need of reason. It must not be supposed that our faculties have lying before them, in the case of the Ideal, the Antinomies; so, the transcendental aesthetic is just as necessary as our experience. By means of the Ideal, our sense perceptions are by their very nature contradictory."
        }
    return cell
    }
}

最佳答案

正如@A-Live 在评论中指出的那样,问题在于嵌套的 ScrollView 。

为了使这个问题对任何人都有用,我最终将 UITextView 替换为 TTTAttributedLabel这显着提高了性能。

此外,无法弄清楚为什么 iOS 9 更新后性能下降如此显着;我怀疑对自动布局引擎进行了未记录的更改;有了一个好的解决方法,就没有必要进一步调查了。

关于ios - 自 iOS 9 以来 UITableView 的滚动性能不佳(之前工作得很好),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34809002/

相关文章:

ios - 在 iOS 中集成 FCM 以进行推送通知时,是否使用了 Device Token?

ios - 使用 .childAdded 类型的 Firebase observe 每次都会检索我的所有信息。请协助

ios - 重新验证用户凭据 Swift

ios - iPhone 无法调用 tableView 函数

iphone - 返回 tableview 后崩溃

ios 6 - 数据加载后立即触发

ios - Storyboard如何处理表格单元格的两个 segue?

ios - 使用 Parse API 将用户名设置为 Facebook 名称。

swift - 通过 Swift 中的联系电话获取数据?

ios - UITableView 在 "pull down to refresh"期间顶部有额外空间