删除UITableView中的行需要花费太多时间

时间:2017-05-17 14:17:22

标签: ios swift uitableview

我正在编写一个包含40k(40000)项数组的程序,并将其显示在UITableView中,之后搜索表应该过滤并仅显示搜索结果。

问题是一次删除多行(例如30000+)大约需要10到20秒,并确保无法使用。你能建议这个问题的决定吗?

tableview.reloadData()不适合)

var allProducts = [Product]()
@IBOutlet weak var searchTableView: UITableView!
@IBOutlet weak var searchTextField: UITextField?

var searchResults = [Product]()

enum Action{
    case Insert
    case Ignore
    case Remove
}

override func viewDidLoad() {
    super.viewDidLoad()
    searchTextField?.addTarget(self, action: #selector(ViewController.textFieldDidChanged(_:)), for: .editingChanged)
    DBBrain().getAllAlcProducts() { [weak self] products in
        self?.allProducts = products
    }
}

func textFieldDidChanged(_ sender: UITextField){
    DispatchQueue.global(qos: .userInitiated).async { [weak self] in
        let text = sender.text!.lowercased()
        let res = self!.allProducts.filter({ $0.name.lowercased().contains(text) })
        if self?.searchTextField?.text != nil && text == self!.searchTextField!.text!{
            if let values = self?.getIndexes(forResults: res){
                self?.searchResults = res
                self?.updateTable(action: values.0, indexes: values.1)
            }
        }
    }
}

private func getIndexes(forResults products: [Product]) -> (Action, [IndexPath]){
    var indexes = [IndexPath]()
    var action = Action.Ignore
    if searchResults.count > products.count{
        var newCounter = 0
        for x in 0..<searchResults.count {
            if products.isEmpty || searchResults[x].id != products[newCounter].id {
                indexes.append(IndexPath(row: x, section: 0))
            }else {
                if newCounter < products.count - 1{
                    newCounter += 1
                }
            }
        }
        action = Action.Remove
    }else if searchResults.count < products.count{
        var oldCounter = 0
        for x in 0..<products.count {
            if searchResults.isEmpty || searchResults[oldCounter].id != products[x].id {
                indexes.append(IndexPath(row: x, section: 0))
            }else {
                if oldCounter < searchResults.count - 1 {
                    oldCounter += 1
                }
            }
        }
        action = Action.Insert
    }
    return (action, indexes)
}

private func updateTable(action: Action, indexes: [IndexPath]) {
    DispatchQueue.main.async { [weak self] in
        if action != .Ignore {
            if action == .Remove {
                self?.searchTableView.beginUpdates()
                self?.searchTableView.deleteRows(at: indexes, with: .fade)
                self?.searchTableView.endUpdates()
            }else if action == .Insert {
                self?.searchTableView.beginUpdates()
                self?.searchTableView.insertRows(at: indexes, with: .fade)
                self?.searchTableView.endUpdates()
            }
        }
    }
}


func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
    return searchResults.count
}

1 个答案:

答案 0 :(得分:0)

你永远不应该有一个包含30k +单元格的表格。这是一种荒谬的记忆用法。您应该为rowsForSection委托方法中的单元格数设置最大限制。

获取结果的计数,如果计数大于30(即使这是一个很大的数字),请将其设置为30.

它也可能拥有超大阵列。这就是大量的内存使用,即使你没有为那些人制作单元格。