【问题标题】:NSUserDefault synchronize on back button swiftNSUserDefault 同步后退按钮 swift
【发布时间】:2016-05-19 08:35:22
【问题描述】:

我的 UITableView 有问题。我有 3 个带有导航控制器的视图。第二个有一个表,第三个是带有一些值的搜索。我想单击按钮搜索(第三个视图)并打开第二个视图并更新表格。但不更新。要工作,我必须返回第一个视图并再次打开第二个视图。

第二次查看代码:

class Empresa: UIViewController, UITableViewDataSource, UITableViewDelegate, UISearchBarDelegate{

let textCellIdentifier = "cell"
var menu:[[String]] = [[]]
var buscaEmp:BuscadorEmpresa = BuscadorEmpresa()


@IBOutlet weak var tablaVista: UITableView!


override func viewDidLoad() {
    super.viewDidLoad()
    recuperaEmpresas()
    tablaVista.delegate = self
    tablaVista.dataSource = self

}

func recuperaEmpresas(){

    menu = buscaEmp.getEmpresas()

}

override func didReceiveMemoryWarning() {
    super.didReceiveMemoryWarning()
    // Dispose of any resources that can be recreated.
}

func numberOfSectionsInTableView(tableView: UITableView) -> Int {
    return 1
}

func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {

    return menu.count
}




func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
    let cell = tableView.dequeueReusableCellWithIdentifier(textCellIdentifier, forIndexPath: indexPath) as! CustomTableViewCell

    let row = indexPath.row
    cell.nombreEmp.text = menu[row][0]
    return cell

}

override func viewWillAppear(animated: Bool) {
    super.viewWillAppear(animated)
    recuperaEmpresas()
    self.tablaVista.reloadData()

}

如您所见,我在 viewWillAppear 上重新加载了我的 tableView (tablaVista) 但不更新。

RecuperaEmpresa()调用这个方法:

 func getEmpresas() -> [[String]]{
    NSUserDefaults.standardUserDefaults().synchronize()

    var array = [[""]]
     if((NSUserDefaults.standardUserDefaults().arrayForKey("ARRAYEMPRESA")) != nil){
array = NSUserDefaults.standardUserDefaults().arrayForKey("ARRAYEMPRESA")! as! [[String]]
return array
}
return array }

我在调试,就像getEmpresa第一次不要更新

编辑:

GetEmpresa 第一次不返回最后一个值,但如果我打印返回正确的值。问题一定是NSUSERDEFAULT

用图像编辑

为亚历山德罗·奥纳诺编辑

最后我的 tableView 重新加载,但我立即看到我的最后一个值并用新闻改变它。我用那个:

第二个视图:

let textCellIdentifier = "cell"
var menu:[[String]] = [[]]
//var buscaEmp:BuscadorEmpresa = BuscadorEmpresa()


@IBOutlet weak var tablaVista: UITableView!


override func viewDidLoad() {

    super.viewDidLoad()
    recuperaEmpresas()
    tablaVista.delegate = self
    tablaVista.dataSource = self

}

func recuperaEmpresas(){
    self.menu = NSUserDefaults.standardUserDefaults().arrayForKey("ARRAYEMPRESA")! as! [[String]]
    //self.menu = buscaEmp.getEmpresas()

}

override func didReceiveMemoryWarning() {
    super.didReceiveMemoryWarning()
    // Dispose of any resources that can be recreated.
}

func numberOfSectionsInTableView(tableView: UITableView) -> Int {
    return 1
}

func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {

    return menu.count
}




func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
    let cell = tableView.dequeueReusableCellWithIdentifier(textCellIdentifier, forIndexPath: indexPath) as! CustomTableViewCell

    let row = indexPath.row
    cell.nombreEmp.text = menu[row][0]
    return cell

}

override func viewDidAppear(animated: Bool) {
    super.viewDidAppear(animated)
    recuperaEmpresas()
    viewDidLoad()
    self.tablaVista.reloadData()

}

第三视图按钮调用此方法:

request.HTTPBody = postString.dataUsingEncoding(NSUTF8StringEncoding);
    let task = NSURLSession.sharedSession().dataTaskWithRequest(request) {
        data, response, error in
        if error != nil
        {
            print("error=\(error)")
            return
        }

        let responseString = NSString(data: data!, encoding: NSUTF8StringEncoding)
        var err: NSError?
        do{
            let myJSON = try NSJSONSerialization.JSONObjectWithData(data!, options: .AllowFragments)

            let jsonEmpresas = myJSON["TablaEmp"]!!["item"]
            let empresas: [[String: AnyObject]]

            if let multipleEmpresas = jsonEmpresas as? [[String: AnyObject]] {
                empresas = multipleEmpresas
            } else if let singleEmpresa = jsonEmpresas as? [String: AnyObject] {
                empresas = [singleEmpresa]
            } else {
                empresas = []
            }

            for empresa in empresas{
                let zcif = empresa["Zcif"] as? String
                let zcccp = empresa["Zcccp"] as? String
                let zfax = empresa["Zfax"] as? String
                let zdocu = empresa["Zdocu"] as? String
                self.arrayEmpresas.append([zcif!, zcccp!, zfax!, zdocu!])


            }
            NSUserDefaults.standardUserDefaults().setObject(self.arrayEmpresas, forKey:"ARRAYEMPRESA")
            NSUserDefaults.standardUserDefaults().synchronize()


        }catch { print(error)}
    }
    task.resume()

还有

 func back(){
    navigationController?.popViewControllerAnimated(true)
}

获取方法:

func getEmpresas() -> [[String]]{
    NSUserDefaults.standardUserDefaults().synchronize()

    var array = [[""]]

    if((NSUserDefaults.standardUserDefaults().arrayForKey("ARRAYEMPRESA")) != nil){
    array = NSUserDefaults.standardUserDefaults().arrayForKey("ARRAYEMPRESA")! as! [[String]]
    return array
    }
    return array
}

【问题讨论】:

  • 您是说要显示在TableView 中的数据是从第二个ViewController 本身修改的,并且更改不会在TableView 中更新?用户默认值在哪里修改?
  • 在 thirdView 的 viewWillDisappear: 方法中更新 serachValue 并在 secondView 的 viewDidAppear: 方法中重新加载您的表格
  • @TheTiger 我试过但还是不行

标签: ios swift tableview


【解决方案1】:

当您多次遇到此类问题时,会涉及采样数据和 UI 更新之间的同步。在你的代码中,我不知道你从哪里得到这些数据,需要多长时间才能到达以及你如何在 NSUserDefaults 上设置这个数组。因此,如果您无法立即看到您的 UI 更新,则可能是您过早启动 reloadData(主线程),因为后台线程仍有操作(http 服务器查询、数据获取......)

你也可以写一个更优雅的:

func getEmpresas() -> [String] {
    NSUserDefaults.standardUserDefaults().synchronize()
    if let array = NSUserDefaults.standardUserDefaults().arrayForKey("ARRAYEMPRESA")! as! [[String]] {
                    return array
    }
    return [String]()
}

更新: 我看过你的更新代码。我想把你的注意力集中在这个方法上:

func recuperaEmpresas(){
    // Are my data ready to read??? who can say to me this in main thread?? 
    menu = buscaEmp.getEmpresas()
}

所以你不能打电话给self.tablaVista.reloadData(),直到你很确定你的数据可以存在。为此,您可以使用completion handler(如果您想知道如何构建它,请点击我)

func recuperaEmpresas(url: NSURL,completionHandler: CompletionHandler){
        callMyNetwork() { //do call and after finish do...
           menu = buscaEmp.getEmpresas()
           self.tablaVista.reloadData()
        }
}

【讨论】:

  • 这不是错误,因为结果类似于 [["123","456"], ["789","000"]] 是数组数组
  • 你能添加一些关于 http 数据获取的代码,以及在你拥有这些数据后你在哪里设置 NSUserDefaults 吗?
  • 已更新(为 Alessandro Ornano 编辑)。谢谢
  • 我正在阅读有关完成处理程序的内容并且有一些问题(这对我来说真的很难)。1- callMyNetwork() 是什么? 2-我必须声明 typealias CompletionHandler = (success:Bool) -> Void 对吗? 3- 当我在 viewDidLoad 和 viewDidAppear 上调用方法时,我必须发送一些参数,但不明白为什么。
  • 不必在任何地方调用你的网络调用方法(调用我的网络),例如你可以只在 viewDidAppear 中使用它,完成后你必须刷新 ui
【解决方案2】:

尝试将 tableVista.reloadData() 放入您的 prepareForSegueunwindToHome 函数中(无论您使用哪个返回到第二个视图)。如果这不可行,您始终可以设置一个 bool var 来确定视图是从第一个视图还是第二个视图加载,然后相应地重新加载数据。

希望这会有所帮助。 :)

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多