我正在使用 UITableView
显示分页列表,我正在使用 UIRefreshControl
表示正在发生网络请求。
但是,由于某种原因, self.refreshControl.beginRefreshing()
/ self.refreshControl.endRefreshing()
会在os_unfair_lock_lock(_lock)
上崩溃。如果我评论这两行,则代码正常。
以下代码可用于复制该问题:
import UIKit
import ReactiveSwift
import Result
class ViewController: UIViewController {
@IBOutlet var tableView : UITableView!
var refreshControl : UIRefreshControl!
var i : Int = 0
var action : Action<Int,[String],NoError>!
var words = [String]()
override func viewDidLoad() {
super.viewDidLoad()
tableView.delegate = self
tableView.dataSource = self
refreshControl = UIRefreshControl()
tableView.addSubview(refreshControl)
action = Action<Int,[String],NoError>{ pageNumber in
return SignalProducer{ sink,_ in
let deadlineTime = DispatchTime.now() + .seconds(2)
DispatchQueue.main.asyncAfter(deadline: deadlineTime) {
if pageNumber == 1{
sink.send(value: ["Apple","Bananas","Clementines"])
}else if pageNumber == 2{
sink.send(value: ["Dodo","Eels","French"])
}
sink.sendCompleted()
}
}
}
action.values.observeValues { [weak self](moreWords) in
if self?.words.isEmpty ?? true {
self?.words = moreWords
}else{
self?.words.append(contentsOf: moreWords)
}
self?.tableView?.reloadData()
}
action.isExecuting.signal.observe(on: UIScheduler()).observeValues { [weak self](loading) in
if loading{
self?.refreshControl.beginRefreshing()
}else{
self?.refreshControl.endRefreshing()
}
print("loading (loading)")
}
}
}
extension ViewController : UITableViewDataSource {
public func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int{
return self.words.count
}
public func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell{
let identifier = "Whatever"
var cell = tableView.dequeueReusableCell(withIdentifier: identifier)
if cell == nil{
cell = UITableViewCell(style: .default, reuseIdentifier: identifier)
}
cell!.textLabel?.text = self.words[indexPath.row]
return cell!
}
}
extension ViewController : UITableViewDelegate{
func scrollViewDidScroll(_ scrollView: UIScrollView) {
if (((scrollView.contentOffset.y + scrollView.frame.size.height) > scrollView.contentSize.height ) ){
if !self.action.isExecuting.value {
i += 1
self.action.apply(i).start()
}
}
}
}
使用 beginRefreshing
&amp;主线程上的endRefreshing
AS:
QueueScheduler.main {
if loading {
self?.refreshControl.beginRefreshing()
} else {
self?.refreshControl.endRefreshing()
}
}
在我的情况下,我也在undairlock中崩溃了,但根本原因是不同的。我有一个类型:
foo.bar <~ foo.bar.producer.skip(first: 1)
实际上,我假设了类似:
self.bar <~ foo.bar.producer.skip(first: 1)
因此,尝试在您的代码中找到自我结合。
这是由种族条件引起的。使用QueueScheduler.main
为我解决了问题:
SignalProducer
.combineLatest(...)
.observe(on: QueueScheduler.main)