ios - 将保存的上下文从 AppDelegate 加载到另一个 ViewController

标签 ios json swift core-data

我正在尝试从我的 Viewcontroller 中创建一个等于我的核心数据已保存的对象的数组。我正在使用核心数据并创建了一个名为 Pokemon 的实体,它具有 3 个属性 nameid一代。在应用程序委托(delegate)中,我使用以下函数从 API 中获取口袋妖怪.这是我解析数据并保存上下文的方法:

 typealias DownloadCompleted = () -> ()
 var pokemonId: Int16 = 0

func fetchPokemon(url: String, completed: @escaping DownloadCompleted) {
    let context = coreData.persistentContainer.viewContext

    let url = URLRequest(url: URL(string: url)!)

    let task = URLSession.shared.dataTask(with: url) { (data, repsonse, error) in
        if error != nil {
            print(error!)
        }

        do {

            let jsonResult = try JSONSerialization.jsonObject(with: data!, options: .mutableContainers) as! NSDictionary
            let jsonArray = jsonResult.value(forKey: "results") as! [[String: Any]]
            for pokemonData in jsonArray {
                self.pokemonId += 1

                if self.pokemonId > 721 {

                    self.coreData.saveContext()
                    return
                }

                guard let name = pokemonData["name"] as? String else {
                    return
                }

                let pokemon = Pokemon(context: context)
                pokemon.name = name
                pokemon.id = self.pokemonId
                print("Name: \(pokemon.name) Id:\(self.pokemonId)")

                if self.pokemonId <= 151 {
                    pokemon.generation = 1
                } else if self.pokemonId <= 251 {
                    pokemon.generation = 2
                }  else if self.pokemonId <= 386 {
                    pokemon.generation = 3
                } else if self.pokemonId <= 493 {
                    pokemon.generation = 4
                } else if self.pokemonId <= 649 {
                    pokemon.generation = 5
                } else if self.pokemonId <= 721 {
                    pokemon.generation = 6
                }

            }



            guard let nextURL = jsonResult.value(forKey: "next") as? String else {
                self.coreData.saveContext()
                return
            }

            DispatchQueue.main.async {
                self.fetchPokemon(url: nextURL, completed: { 
                    self.coreData.saveContext()
                })
                completed()
            }

        } catch let err {
            print(err.localizedDescription)
        }



    }
    task.resume()

}

这就是我在 appDelegate 中的调用方式。真的不知道在 fetchPokemon 中间要做什么或如何在另一个 View Controller 中调用它。所以我将其留空,不确定这是否与我遇到的问题有关。

func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool {

    let context = self.coreData.persistentContainer.viewContext
    let pokemonListVC = self.window?.rootViewController as! PokemonListVC
    pokemonListVC.context = context
    fetchPokemon(url: pokemonAPI) {

    }
    return true
}

我正在使用应用商店中的这个 SQL-Light 只读应用。我检查了数据,所有 721 只神奇宝贝都在保存。现在,我不知道如何才能使我的 View Controller 中的数组等于所有保存的 721 个口袋妖怪。我将此代码添加到我的 viewController 中。

class PokemonListVC: UIViewController {

  weak var context: NSManagedObjectContext! {
      didSet {
        return pokemon = Pokemon(context: context)
      }
  }
  var pokemon: Pokemon? = nil
  lazy var pokemons = [Pokemon]()
  override func viewDidLoad() {
      super.viewDidLoad()
      loadData()
  }

  func loadData() {
    pokemons = pokemon!.loadPokemon(generation: 1, context: context)
  }
}

我创建了我的 Pokemon 实体的扩展,并添加了一个函数 loadPokemon 来按代过滤 Pokemon。这是代码。

extension Pokemon {

func loadPokemon(generation: Int16 = 0, context: NSManagedObjectContext) -> [Pokemon] {
    let request: NSFetchRequest<Pokemon> = Pokemon.fetchRequest()
        request.predicate = NSPredicate(format: "generation = %@", generation)
        request.sortDescriptors = [NSSortDescriptor(key: "name", ascending: true)]
        do {
            let pokemons = try context.fetch(request)
            print("My Pokemon count: \(pokemons.count)")
            return pokemons
        } catch let err {
            print(err.localizedDescription)
        }
    return []
}

}

当我在我的 ViewController 中调用 loadData 时,它崩溃了。数组计数为 0,英雄扩展中的计数也是如此。所以我不知道如何使我的数组等于从 coreData 保存的神奇宝贝。

非常感谢提供的任何帮助。 :)

这是我的 deleteRecords 代码,它也在我的 appDelegate 中。这会在应用程序启动时删除所有记录。我在 fetchPokemons 之前的 didFinishLaunchingWithOption 函数的开头调用了这个方法。

func deleteRecords() {
    let context = coreData.persistentContainer.viewContext
    let pokemonRequest: NSFetchRequest<Pokemon> = Pokemon.fetchRequest()

    var deleteRequest: NSBatchDeleteRequest
    var deleteResults: NSPersistentStoreResult

    do {
        deleteRequest = NSBatchDeleteRequest(fetchRequest: pokemonRequest as! NSFetchRequest<NSFetchRequestResult>)
        deleteResults = try context.execute(deleteRequest)
    } catch let err {
        print(err.localizedDescription)
    }

}

最佳答案

正如您所说,您已确定所有 pockemon 记录都正确存储在您的 coredata 中,您可以通过提供获取请求来简单地从您的代码数据中获取记录。我已经为联系人存储创建了演示,我可以通过这个获取请求获取所有联系人,你可以在你想要获取所有记录的 ViewController 中尝试这段代码。

 let appDelegate = UIApplication.shared.delegate as! AppDelegate
    let managedContext = appDelegate.persistentContainer.viewContext
    let fetchRequest = NSFetchRequest<NSManagedObject> (entityName: "Pokemon")

    do {
        arrPockemon = try managedContext.fetch(fetchRequest)

    }catch let error as NSError {
        showAlert(string: error.localizedDescription)
    }

首先尝试获取所有记录,如果获取了所有记录,则开始过滤扩展名和所有记录。希望它能帮助你。你可以从这里学习https://code.tutsplus.com/tutorials/core-data-and-swift-core-data-stack--cms-25065

在 userDefault 上保存标志。

 //check for first time when app is installed first time(first time flag is not present so)
    let userDefault = UserDefaults.standard.dictionaryRepresentation()
    if userDefault.keys.contains("isDataAvailable") {
        //key is availebe so check it
        if userDefault["isDataAvailable"] as! String == "1"{
            //no need to call server for data
        }else{
            //fetch data from server
            // once you get data from server make isDataAvailable flage as 1
            UserDefaults.standard.setValue("1", forKey: "isDataAvailable")
            UserDefaults.standard.synchronize()
        }
    }
    else{
        //flag is not avalable so call server for data
        // once you get data from server make isDataAvailable flage as 1
        UserDefaults.standard.setValue("1", forKey: "isDataAvailable")
        UserDefaults.standard.synchronize()
    }

关于ios - 将保存的上下文从 AppDelegate 加载到另一个 ViewController,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43290506/

相关文章:

json - Go Lang 帮助 - 访问数组/接口(interface) slice

JSON.parse() 意外 token 错误

iOS 8 Swift CS193P 计算器 Bug 开关操作

swift - 在 Swift 中使用 Sprite Kit 呈现场景

ios - 使用 UIScrollView 更改图像的 alpha

ios - Swift Table View Cell 布局折叠所有元素

java - JSON 字符串到对象列表

swift - 当用户在另一个应用程序中返回到我的应用程序时刷新 UIViewController

ios - UILongPressGestureRecognizer 在状态改变时继续动画

ios - Braintree Apple sandbox Pay 在扫描指纹 iOS 后显示付款未完成错误