Swift ForEach JSON 词典的问题

Swift ForEach Issues with JSON Dictionary

我正在尝试构建一个从 API 端点接收 JSON 对象的应用程序,然后我想在视图中列出该对象。我看过很多关于这个主题的视频,但在每个视频中,他们都使用非常简单的 JSON 对象作为示例,因此他们编写的代码似乎并没有真正转移过来,无论我如何给我错误尝试格式化它。代码如下

import SwiftUI
import Combine
import Foundation

public struct ActivityModel: Codable, Identifiable {
    public let id: Int
    public let name: String
    public let activity_desc: String?
}


public struct ActivitiesModel2: Codable {
    public let location: String
    public let popular: [String:ActivityModel]
}

public struct ActivitiesModel: Codable {
    public let activities: ActivitiesModel2
}

public class ActivityFetcher: ObservableObject {
    var activities: ActivitiesModel?

    init(){
        guard let url = URL(string: "https://mywebsite.com/api/loadapi") else { return }

        var urlRequest = URLRequest(url: url)
        urlRequest.httpMethod = "POST"

        URLSession.shared.dataTask(with: urlRequest) { (data, response, error) in

            do {
                if let d = data {
                    let decodedLists = try JSONDecoder().decode(ActivitiesModel.self, from: d)
                    DispatchQueue.main.async {
                        self.activities = decodedLists
                    }
                } else {
                    print("No Data")
                }
            } catch {
                print("Error")
            }

        }.resume()
    }
}

struct ActivityGuestView: View {
    let networkingServiceGeneral = NetworkingServiceGeneral()

    @ObservedObject var viewRouter: ViewRouter

    @ObservedObject var fetcher = ActivityFetcher()

    var body: some View {
        // This is where my issues start
        List(fetcher.activities?.activities.popular) { result in
            VStack {
                Text(result.name)
                Text(result.activity_desc)
                    .font(.system(size: 11))
                    .foregroundColor(Color.gray)
            }
        }
    }
}

正如我所说,这段代码给出了 5 个错误。他们是以下;

    - Initializer 'init(_:rowContent:)' requires that '(key: String, value: ActivityModel)' conform to Identifiable
    - Initializer 'init(_:rowContent:)' requires that '[String : ActivityModel]' conform to 'RandomAccessCollection'
    -Value of optional type '[String : ActivityModel]?' must be unwrapped to a value of type '[String : ActivityModel]'
    - Coalesce using '??' to provide a default when the optional value contains 'nil'
    - Force-unwrap using '!' to abort execution if the optional value contains 'nil'

其中一些错误有修复它的选项,但是当我按下修复时它会添加代码但实际上并没有修复错误所以我想无论如何都包含它们。我对 Swift 还是很陌生,但我知道其中的一些要求,特别是符合 Identifiable 的要求,但是当我尝试添加标签时,它说 struct ActivitiesModel 不符合 identifiable,并且JSON 对象没有该部分的 ID,因此我无法请求 ID 使其可识别。

任何帮助将不胜感激,这现在有点像一堵墙。

这是JSON

"activities": {
        "location": "Dallas",
        "popular": {
            "10": {
                "id": 38,
                "name": "Adventure Landing Dallas",
                "activity_desc": "Aquatic complex chain with additional land attractions including mini-golf, laser tag & go-karts.",
            },
            "12": {
                "id": 40,
                "name": "Jumpstreet",
                "activity_desc": "None provided.",
            },
        }
    }
}

你有几个问题。

首先,字典和 List 不兼容;你真的需要一个数组。正如错误所说,提供给 List 的项目需要确认 IdentifiableRandomAccessCollection。字典不符合任何一个,你不能真的让它这样做。

你的第二个问题是 fetcher.activities 是可选的,List 初始化器不能接受可选的。编译器建议了几个替代方案 - 使用 nil 合并运算符 (??) 提供默认值或使用 ! 强制解包(这会崩溃,因为你知道 fetcher.activities 正在运行最初是 nil

第一个问题的根本原因是您的 JSON 具有可变键而不是简单的活动数组。这不是一个好主意,你真的应该在服务器端做一些工作来消除无意义的数字键并有一个简单的数组,根据我的评论,但是你已经表明你不能这样做。

鉴于此,另一种方法是使用计算 属性:

将字典值公开为数组
public struct ActivitiesModel2: Codable {
    public let location: String
    private var popular: [String:ActivityModel] 
    public var popularActivities: [ActivityModel] {
        get {
            return Array(self.popular.values)
        }
    }
}

虽然我们在这里,但我们可以用 CodingKeys 枚举修复 un-Swifty activity_desc

public struct ActivityModel: Codable, Identifiable {
    public let id: Int
    public let name: String
    public let activityDesc: String?

    enum CodingKeys: String, CodingKey {
        case id = "id"
        case name = "name"
        case activityDesc = "activity_desc"
    }
}

现在,您可以重新编写视图以使用新的计算 属性 并处理可选的:

struct ActivityGuestView: View {
    let networkingServiceGeneral = NetworkingServiceGeneral()

    @ObservedObject var viewRouter: ViewRouter

    @ObservedObject var fetcher = ActivityFetcher()

    var body: some View {
        // This is where my issues start
        List(fetcher.activities?.activities.popularActivities ?? []) { result in
            VStack {
                Text(result.name)
                Text(result.activity_desc)
                    .font(.system(size: 11))
                    .foregroundColor(Color.gray)
            }
        }
    }
}