ios - CoreData 完成后 Swift iOS 9 UITableView 重新加载数据

标签 ios swift uitableview asynchronous reloaddata

我尝试在 REST 调用完成并且所有实体都存储在 CoreData 中后更新我的 TableView 。我尝试了dispatch_asynch,但 TableView 没有按预期重新加载。我正在使用 UIRefreshControl 来触发刷新。通常,刷新2次后即可显示正确的tableview数据。我不知道为什么。

我可以确认我的 tableview 实例不为零。

@IBAction func refreshTriggered(sender: UIRefreshControl) {

    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), {
        self.evMan.retrieveAndSaveEvents(self.pubMan.getAllPublishers())


      dispatch_async(dispatch_get_main_queue()) {
          self.tableView.reloadData()
          self.refreshControl?.endRefreshing()
      }
   }
}

这是我的“EventManager”evMan 中的retrieveAndSaveEvents 方法:

func retrieveAndSaveEvents(forPublishers: [PUBLISHERS]) {

    for publisher in forPublishers {

        let pubId = publisher.id as Int
        let str = "/publishers/\(pubId)/events"

        // resty is an instance of my REST-api wrapper
        self.resty.GET(self.server, route: str, onCompletion: {json in

            let result = json.asArray

            for var i = 0; i < result!.count; i++ {

                if !self.isIDAlreadyInDB(json[i]["id"].asInt!) {
                    let newEv = NSEntityDescription.insertNewObjectForEntityForName("EVENTS", inManagedObjectContext: self.context!) as! EVENTS
                    newEv.headline = json[i]["headline"].asString!
                    newEv.id = json[i]["id"].asInt!
                    newEv.content = json[i]["content"].asString!
                    newEv.ofPublisher = publisher

                    do {
                        try self.context!.save()
                    } catch _ {
                    }

                    print("New Event from \(publisher.name): \(newEv.headline)")
                }
            }
        })
    }
}

仅供引用,这是我的 cellForRowAtIndexPath:(我正在使用自定义 UITableViewCell)

override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {

    let cell = self.tableView.dequeueReusableCellWithIdentifier("eventCell") as? EventCell

    let curEv = evMan.getEvents()[indexPath.row]

    cell?.infoLabel.text = curEv.ofPublisher?.name
    cell?.myImageView.image = UIImage(named: "icon.png")
    cell?.detailLabel.text = curEv.headline
    cell?.timeLabel.attributedText = NSAttributedString(string: self.dateFormatter.stringFromDate(curEv.updatedAt))
    cell?.contentView.backgroundColor = UIColor.clearColor()
    cell?.backgroundColor = UIColor(white: 1.0, alpha: 0.5)

    return cell!
}

这是我的 REST 包装器:

class REST: NSObject {

// Basic Auth
let sConfig = ServerConfig()

var username = "rest"
var password = "resttest"

override init() {
    username = sConfig.getUserLogin().user
    password = sConfig.getUserLogin().pass
}

func GET(server: String, route: String, onCompletion: (JSON) -> Void) {
    let route = server+route
    makeHTTPGetRequest(route, onCompletion: { json, err in
        onCompletion(json as JSON)
    })
}

func makeHTTPGetRequest(path: String, onCompletion: ServiceResponse) {
    let request = NSMutableURLRequest(URL: NSURL(string: path)!)

    let loginString = NSString(format: "%@:%@", username, password)
    let loginData: NSData = loginString.dataUsingEncoding(NSUTF8StringEncoding)!
    let base64LoginString = loginData.base64EncodedStringWithOptions(NSDataBase64EncodingOptions())
    let loginvalue = "Basic " + base64LoginString

    // add Headers
    request.addValue(loginvalue, forHTTPHeaderField: "Authorization")
    request.addValue("application/json", forHTTPHeaderField: "Accept")
    request.addValue("application/json", forHTTPHeaderField: "Content-Type")

    let session = NSURLSession.sharedSession()

    let task = session.dataTaskWithRequest(request, completionHandler: {data, response, error -> Void in
        let json:JSON = JSON(data: data!)
        onCompletion(json, error)
    })
    task.resume()
}
}

最佳答案

@IBAction func refreshTriggered(sender: UIRefreshControl) {

    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), {
        self.evMan.retrieveAndSaveEvents(self.pubMan.getAllPublishers())
    })

    dispatch_async(dispatch_get_main_queue()) {
        self.tableView.reloadData()
        self.refreshControl?.endRefreshing()
    }
}

现在,您不能保证 reloadData()self.evMan.retrieveAndSaveEvents(self.pubMan.getAllPublishers()) 之后发生,因为它们不会在同一时间发生队列。很可能,reloadData() 发生在检索之前。

选项 1:

你应该放置 block :

   dispatch_async(dispatch_get_main_queue()) {
        self.tableView.reloadData()
        self.refreshControl?.endRefreshing()
    }

funcretrieveAndSaveEvents(self.pubMan.getAllPublishers())的末尾

选项 2:

 @IBAction func refreshTriggered(sender: UIRefreshControl) {

        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), {
            self.evMan.retrieveAndSaveEvents(self.pubMan.getAllPublishers())
            dispatch_async(dispatch_get_main_queue()) {
                self.tableView.reloadData()
                self.refreshControl?.endRefreshing()
            }
        })

    }
<小时/>

编辑:

我没有看到retrieveAndSaveEvents中有另一个队列

所以,把

   dispatch_async(dispatch_get_main_queue()) {
        self.tableView.reloadData()
        self.refreshControl?.endRefreshing()
    }

最后:

self.resty.GET(self.server, route: str, onCompletion: {json in
        let result = json.asArray

        for var i = 0; i < result!.count; i++ {

            if !self.isIDAlreadyInDB(json[i]["id"].asInt!) {
                let newEv = NSEntityDescription.insertNewObjectForEntityForName("EVENTS", inManagedObjectContext: self.context!) as! EVENTS
                newEv.headline = json[i]["headline"].asString!
                newEv.id = json[i]["id"].asInt!
                newEv.content = json[i]["content"].asString!
                newEv.ofPublisher = publisher

                do {
                    try self.context!.save()
                } catch _ {
                }

                print("New Event from \(publisher.name): \(newEv.headline)")
            }
        }
    })

关于ios - CoreData 完成后 Swift iOS 9 UITableView 重新加载数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32122929/

相关文章:

ios - Apple Pay - 通行证为空

swift - truncatingRemainder 与 Swift 中的余数

ios - 无法保存复选标记以显示在 UITableView 上

带 Chromecast 的 Swift 框架 : include of non-modular header inside framework module

webview - 在 Swift 的 webview 上启用 web inspector

ios - 使用 firebase 中添加的 child 从 firebase 获取数据

ios - UITableViewCell 自动调整高度不工作

ios - 我怎样才能使应用程序图标像 iOS 内置的时钟和日历图标一样响应

ios - 将 XLPagerTabStrip 转换为 swift3 会导致错误

ios - 方法不是公认的客观 c 方法