gpt4 book ai didi

ios - 将 API 数据存储到 UserDefaults 并打印到列表

转载 作者:行者123 更新时间:2023-12-01 19:30:19 26 4
gpt4 key购买 nike

在我的应用程序中,用户扫描条形码并从 API 获取有关产品的信息。
我想创建一个历史部分,用户可以在其中查看最近的 10 个产品。
API 数据的结果存储在 Result 类型中,为了能够显示在列表中,它必须是可识别的。
结果是我用来存储来自 API 调用的产品详细信息的自定义数据类型。
结果

struct Result: Codable, Identifiable {
var id = UUID()
var description: String?
var brand: String?
var ingredients: String?
var image: String?
var upc_code: String?
var return_message: String?
var return_code: String?

enum CodingKeys: String, CodingKey {
case description, brand, ingredients, image, upc_code, return_message, return_code
}
}
此数据类型存储我将显示为列表的 Result 数组
历史
struct History: Codable {
var results: [Result]
}
这是 API 调用:
func loadData(url: String, completion: @escaping (Error?, Result?) -> Void ) {
if let url = URL(string: url) {
let task = URLSession.shared.dataTask(with: url) { data, response, error in
guard let data = data, error == nil else {return}

do {
let defaults = UserDefaults.standard
let encoder = JSONEncoder()
if let encoded = try? encoder.encode(data) {
var sizeCheck = defaults.object(forKey:"productHistory") as? [Data] ?? [Data]()
if (sizeCheck.count == 10) { //Check if there's more than 10 products already on the history list
sizeCheck.removeLast()
}
sizeCheck.append(encoded) //Add new product to list
defaults.set(sizeCheck, forKey: "productHistory") //Add new list to userDefaults
}
let decoder = JSONDecoder()
let result: Result = try decoder.decode(Result.self, from: data)
completion(nil, result) //Used elsewhere to display the scanned product after it's been added to the history list
}
catch let e {
print(e)
completion(e, nil)
}
}

task.resume()
}
}
这是我的 View ,当按下按钮时显示列表中的最后 10 个产品。
最后 10 个产品应存储在 UserDefaults 中,键为 productHistory .这是在 API 调用 LoadData() 中完成的
struct historyView: View {
@Binding var showingHistory: Bool
@State private var results = [Result]()

var body: some View {
let defaults = UserDefaults.standard
if let products = defaults.object(forKey: "productHistory") as? Data {
if let decodedResponse = try? JSONDecoder().decode(History.self, from: products) {
self.results = decodedResponse.results
}
}
return List(self.results, id: \.id) { item in
Text(item.description!)
}
}
}
据我了解,问题在于 UserDefaults 无法存储 JSON 数据。因此,当获取 API 数据时,我将数据按原样存储到 userdefualts 中。然后在需要时对其进行解码,例如将其存储在历史记录中或显示它。
目前我得到一个空白列表,下面的 if 语句没有通过。
if let decodedResponse = try? JSONDecoder().decode(History.self, from: products) {
如果我将 URL 粘贴到浏览器中,这是来自 API 的 JSON 数据:

编辑
这是我的 APICall():
func callAPI() -> String {
if (scannedCode.barcode == "") {
return "noneScanned"
}
else {
let hashedValue = scannedCode.barcode.hashedValue("API ID")
//print(hashedValue!)
loadData(url: "URL") { error, result in
if let err = error {
self.APIresult = err.localizedDescription
print(APIresult)
//output error
}
else if (result?.ingredients == nil) {
DispatchQueue.main.async {
self.APIresult = "noIngredients"
}
}
else if (result?.description == nil) {
DispatchQueue.main.async {
self.APIresult = "noDescription"
}
}
else {
DispatchQueue.main.async {
self.APIresult = "success"
}
}
DispatchQueue.main.async {
product.result = result!
//updates view that show's the scanned product, as it's @Published
}
}
return APIresult
}
}
在本节中,我想查找我拥有的有关产品的数据并进行相应处理。因此,使用上述解决方案,我会根据是否有图像或描述等返回不同的值...
使用 vadian 解决方案,我将其更改为:
          loadData(url: "URL") { result  in
switch result {
case .success(product):
print("success")
case .failure(error):
print("failure")
}
}

最佳答案

正如评论中提到的,你混淆了DataResult首先放下History并重命名 ResultProduct .我们将保存 Product 的数组至UserDefaults

struct Product: Codable, Identifiable {
var id = UUID()
var description: String?
var image: String?
var upc_code: String?
var return_message: String?
var return_code: String?

private enum CodingKeys: String, CodingKey {
case description, image, upc_code, return_message, return_code
}
}
loadData使用通用 Result类型作为闭包参数。收到数据后解码为 Product实例,然后加载保存的数组,删除第一个(!)项(如有必要)附加新项,保存数组并使用新的 Product 调用完成.所有潜在的错误都在 failure 中传递。案子。
func loadData(url: String, completion: @escaping (Result<Product,Error>) -> Void ) {
guard let url = URL(string: url) else { return }
let task = URLSession.shared.dataTask(with: url) { data, response, error in
if let error = error { completion(.failure(error)); return }

do {
let decoder = JSONDecoder()
let product = try decoder.decode(Product.self, from: data!)
let defaults = UserDefaults.standard
var history = [Product]()
if let readData = defaults.data(forKey:"productHistory") {
do {
history = try decoder.decode([Product].self, from: readData)
if history.count == 10 { history.removeFirst() }
} catch { print(error) }
}
history.append(product)
let saveData = try JSONEncoder().encode(history)
defaults.set(saveData, forKey: "productHistory")
completion(.success(product))
}
catch {
print(error)
completion(.failure(error))
}
}
task.resume()
}
并称之为
loadData(url: "URL") { result  in
switch result {
case .success(let product):
if product.ingredients == nil {
self.APIresult = "noIngredients"
} else if product.description == nil {
self.APIresult = "noDescription"
} else {
self.APIresult = "success"
}
product.result = product

case .failure(let error):
self.APIresult = error.localizedDescription
print(APIresult)
}
}
HistoryView (请以大写字母命名结构)从 UserDefaults 获取数据并解码 Product大批。
struct HistoryView: View {
@Binding var showingHistory: Bool
@State private var results = [Product]()

var body: some View {
let defaults = UserDefaults.standard
if let historyData = defaults.data(forKey: "productHistory") {
do {
self.results = try JSONDecoder().decode([Product].self, from: historyData)
} catch { print(error) }
}
return List(self.results, id: \.id) { item in
Text(item.description ?? "n/a")
}
}
}
注意:请注意 UUID 没有被编码和保存。
请使用更具描述性的变量名称。

关于ios - 将 API 数据存储到 UserDefaults 并打印到列表,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/63215006/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com