天天看點

【iOS】Runloop優化清單滑動卡頓

此圖是我卡頓的清單頁截屏,為了避免打廣告和洩漏隐私的緣故,部分地方會打馬賽克,望諒解,不過絕對不影響讀者閱讀:

【iOS】Runloop優化清單滑動卡頓

// // JCWFinancialFroductListVC.swift // HengYouCai // // Created by wll on 2017/11/9. // Copyright © 2017年 wll. All rights reserved. // 理财産品清單--代碼

import UIKit

///定義blcok typealias RunloopBlock = ((IndexPath) -> Bool) private let cellID = "XXXFinancialFroductListVCCell"

//MARK:界面初始化 class XXXFinancialFroductListVC: BaseTableViewController {

//iOS11 系統bug會特别卡 ///時鐘事件 var timer: Timer? ///任務數組 var tasksArr: [RunloopBlock] = [] ///任務對應indexPath數組 var tasksIndexPathArr: [IndexPath] = [] ///最大任務 10 let maxQueueLength = XXXLimitNumberPageSize

///上拉下拉重新整理管理器 let pageManager = XXXRefreshPageManager() override func viewWillAppear(_ animated: Bool) { super.viewWillAppear(animated) //每次進來重新整理 if pageManager.pageNo>1 {

let index = IndexPath(row: 0, section: 0) self.tableView.scrollToRow(at: index, at: .top, animated: true) } pageManager.pageNo = 1

loadData()

} override func viewDidLoad() { super.viewDidLoad() self.view.jcw_showGifLoading() ///界面配置 self.createNav() self.createCV()

//添加runloop的監聽 self.addRunloopObserver()

weak var weakSelf = self self.tableView.mj_header = WLRefreshGifHeader(refreshingBlock: { weakSelf?.pageManager.pageNo = XXXLimitNumberStarPage weakSelf?.loadData() }) }

func createNav() { self.navigationItem.title = "理财産品" }

func createCV() {

self.tableView.register(XXXFinancialProductCell.self, forCellReuseIdentifier: cellID) self.tableView.backgroundColor = Normal_458Color self.tableView.separatorStyle = .none self.tableView.estimatedRowHeight = 0 self.tableView.estimatedSectionFooterHeight = 0 self.tableView.estimatedSectionHeaderHeight = 0 ///頂部間隙 let topGrayView = UIView(frame: CGRect(x: 0, y: 0, width: XXXSCREENW, height: 8)) topGrayView.backgroundColor = self.tableView.backgroundColor self.tableView.tableHeaderView = topGrayView } }

//MARK:表代理 extension XXXFinancialFroductListVC { override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { return self.pageManager.dataArr.count }

override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { let cell: XXXFinancialProductCell = tableView.dequeueReusableCell(withIdentifier: cellID) as! XXXFinancialProductCell self.addTask(indexPath) { (indexP) -> Bool in cell.kaDunHandler(self.pageManager.dataArr, tableView, indexP) return true } cell.kaDunHandler2(self.pageManager.dataArr, tableView, indexPath) return cell }

override func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat { return 128 }

override func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) { let final = FinancialDetailVC() let model = pageManager.dataArr[indexPath.row] as! ProjectModel

final.homeModel = model final.idStr = model.id self.navigationController?.pushViewController(final, animated: true, withHideBottomBarWhenPush: true, pop: false) }

}

//MARK:處理卡頓 extension XXXFinancialFroductListVC {

///添加新的任務的方法! func addTask(_ indexP: IndexPath, unit: @escapingRunloopBlock) { self.tasksArr.append(unit) self.tasksIndexPathArr.append(indexP) //判斷一下 保證沒有來得及顯示的cell不會繪制 if self.tasksArr.count > self.maxQueueLength { _ = self.tasksArr.remove(at: 0) _ = self.tasksIndexPathArr.remove(at: 0) } } ///添加runloop的監聽 fileprivate func addRunloopObserver() { //擷取目前RunLoop let runLoop: CFRunLoop = CFRunLoopGetCurrent() //定義一個上下文 var context: CFRunLoopObserverContext = CFRunLoopObserverContext(version: 0, info: unsafeBitCast(self, to: UnsafeMutableRawPointer.self), retain: nil, release: nil, copyDescription: nil) //定義一個觀察者 if let observer = CFRunLoopObserverCreate(kCFAllocatorDefault, CFRunLoopActivity.beforeWaiting.rawValue, true, 0, self.observerCallbackFunc(), &context){ //添加目前RunLoop的觀察者 CFRunLoopAddObserver(runLoop, observer, .commonModes); } } ///回調函數 func observerCallbackFunc() -> CFRunLoopObserverCallBack { return {(observer, activity, context) -> Void in if context == nil {//如果沒有取到 直接傳回 return } // 崩潰的與原因不在這裡 isFromADGuide 的問題 不應該取反 //取出上下文 就是目前的vc let vc = unsafeBitCast(context, to: XXXFinancialFroductListVC.self) //取出任務 if let unit = vc.tasksArr.first, let indexP = vc.tasksIndexPathArr.first { var result = false while (result == false && !vc.tasksArr.isEmpty && !vc.tasksIndexPathArr.isEmpty){ //執行任務 result = unit(indexP); //幹掉第一個任務 _ = vc.tasksArr.remove(at: 0) _ = vc.tasksIndexPathArr.remove(at: 0) } } } } }

//MARK:清單資料請求 extension XXXFinancialFroductListVC { fileprivate func loadData() { XXXNetService.productsList(withDict: ["XXX":"x","pageNo":self.pageManager.pageNo,"pageSize":XXXLimitNumberPageSize], onSuccess: { (result) in

//隐藏加載框 self.view.hideHUDLoading(0)

//坑 一定記得加可選值 否則 footer可能不存在 self.tableView.mj_header?.endRefreshing() self.tableView.mj_footer?.endRefreshing()

let resultList: [Any]? = JSON(result asAny).dictionaryValue["productsList"]?.arrayObject

if self.pageManager.pageNo == XXXLimitNumberStarPage { self.pageManager.dataArr.removeAll() //下拉清除任務 self.tasksArr.removeAll() self.tasksIndexPathArr.removeAll() }

if let count = resultList?.count, count >= XXXLimitNumberPageSize{ self.pageManager.pageNo += XXXLimitNumberStarPage weak var weakSelf = self self.tableView.mj_footer = WLRefreshFooter(refreshingBlock: { weakSelf?.loadData() }) }else{ self.tableView.mj_footer?.endRefreshingWithNoMoreData() } self.pageManager.dataArr.append(contentsOf: ProjectModel .mj_objectArray(withKeyValuesArray: resultList)) self.tableView.reloadData()

}) { (code, errMsg) in //隐藏加載框 self.view.hideHUDLoading(0) self.tableView.mj_header?.endRefreshing() self.tableView.mj_footer?.endRefreshing() } } }