iOS面试题(二) UI视图 -UITableView重用池机制及 数据源同步问题


UI视图

  • UITableView相关(重用机制的理解和运用方面,包括我们在UITableView的使用过程中,会涉及到对数据源的一个多线程的操作,这里面就涉及到了一个数据源同步的问题)

  • 事件响应&视图响应

  • 图像显示原理(UI视图的图像显示原理是怎样的?我们创建了一个控件,最后是如何转化到屏幕上的每一个像素点的呢?)

  • UI卡顿、掉帧

  • UI绘制原理/异步绘制原理

  • 离屏渲染

 

 UITableView相关


 1. 字母索引条(重用机制, 重用字母索引)

            

  (1)  UITableView的重用机制可以理解为四个部分,即将滚出可视区域的cell,当前屏幕显示的cell,即将滚入可视范围内的cell,重用池。

  (2)即将滚入可视区域的cell在滚入之前去缓冲池中根据identifier寻找有没有这个cell,如果没有就创建,有就复用。

  (3)即将滚出可视区域的cell在滚出之后去缓冲池中根据identifier寻找有没有这个cell,如果没有就加入缓冲池。

  (4)如此上下循环滚动根据identifier完成缓冲池的更新和cell的重用

2. 数据源同步问题:删除一条数据(主线程)--> 数据源 < --  LoadMore (子线程)   (涉及到了多线程对共享数据源的一个同步问题,面试的时候可能会被问到 如何解决这种tableView在多线程环境下的去修改或者访问数据源这样的同步问题)
                      
     (1)并发访问、数据拷贝

                               
         ① 比方说现在有主线程和子线程,我们在做数据拷贝的时候,一般在主线程当中。
         ② 拷贝之后呢会把拷贝的结果给子线程来使用,同时在子线程里进行新数据的网络请求、数据解析和预排版等。
         ③ 在子线程操作的同时,我们在主线程删除一行数据(为了防止和子线程数据不同步,需要记录删除操作)、reload UI后,此条数据会消失不见,如果还有时间,主线程会做other work一切其他的操作。
         ④ 子线程将要返回数据,更新主线程UI的时候,同步删除操作,在子线程里面把拷贝过来的数据也做一次删除的操作
         ⑤ 然后回到主线程去reload UI。

  (2)串行访问
                                    
        比方说有主线程和子线程,既然是串行访问,就要使用到GCD当中的一个串行队列。
        ① 此时比如说子线程正在进行网络请求、数据解析,然后它会把网络数据请求过来的一部分在串行队列当中进行新增数据预排版,当然这一过程是在子线程当中进行的。
        ② 如果在这一过程当中,在主线程里面删除某一行数据的话,需要以同步的方式在串行队列当中进行处理,此时如果子线程正在进行新增数据预排版等操作的时候,那么主线程要等一小会Waiting,然后在串行队列当中,前一个block任务执行完成之后,然后去同步主线程发送的任务数据删除。
         ③ 把数据删除同步之后,再回到主线程reload UI 。这样就能保证无论是主线程还是子线程,对tableView的数据源的操作都是在串行队列中进行的,就可以保证数据的一个同步的问题,避免UI刷新错乱的一个现象。

   总结:两种方案各有利弊,比如②串行访问方案,要求在子线程处理任务特别耗时的时候呢,我们某一个删除动作可能有一定的延迟。对于①并发访问、数据拷贝来说呢,有数据同步的一个操作(记录数据删除的一个动作),然后我们还需要大量数据源的一个拷贝,内存的开销是有一定的问题。所以应该根据实际业务选择哪种方案。

 

  • 0
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
iOS中,一个视图只能有一个UITableView。但是可以通过创建多个UITableView来实现一个视图中显示多个表格的效果。以下是一个示例代码: 首先,你需要在视图控制器中添加多个UITableView的实例变量: ```swift class YourViewController: UIViewController { var tableView1: UITableView! var tableView2: UITableView! // ... } ``` 然后,在视图加载完成后,你可以创建和配置这些UITableView的实例: ```swift override func viewDidLoad() { super.viewDidLoad() // 创建第一个UITableView tableView1 = UITableView(frame: CGRect(x: 0, y: 0, width: view.frame.width, height: view.frame.height/2)) tableView1.dataSource = self tableView1.delegate = self view.addSubview(tableView1) // 创建第UITableView tableView2 = UITableView(frame: CGRect(x: 0, y: view.frame.height/2, width: view.frame.width, height: view.frame.height/2)) tableView2.dataSource = self tableView2.delegate = self view.addSubview(tableView2) // ... } ``` 接下来,你需要实现UITableViewDataSource和UITableViewDelegate协议的相关方法来提供表格的数据和处理交互事件。例如: ```swift extension YourViewController: UITableViewDataSource { func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { if tableView == tableView1 { // 返回第一个UITableView的行数 return 10 } else if tableView == tableView2 { // 返回第UITableView的行数 return 5 } return 0 } func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { let cell = tableView.dequeueReusableCell(withIdentifier: "Cell", for: indexPath) if tableView == tableView1 { // 配置第一个UITableView的单元格 cell.textLabel?.text = "Table View 1 - Row \(indexPath.row)" } else if tableView == tableView2 { // 配置第UITableView的单元格 cell.textLabel?.text = "Table View 2 - Row \(indexPath.row)" } return cell } } extension YourViewController: UITableViewDelegate { func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) { if tableView == tableView1 { // 处理第一个UITableView的行选中事件 print("Table View 1 - Row \(indexPath.row) selected") } else if tableView == tableView2 { // 处理第UITableView的行选中事件 print("Table View 2 - Row \(indexPath.row) selected") } } } ``` 这样,你就可以在同一个视图中使用多个UITableView了。记得在视图控制器中遵循UITableViewDataSource和UITableViewDelegate协议,并在视图加载完成后设置数据源和代理。 希望这能帮到你!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值