我正在尝试从IEX finance API中获取库存数据,并且我需要的一切都可以很好地输出到控制台,但是在尝试使用该数据插入表格视图时遇到了问题。

这是API url,这是JSON in a readable format

这些是我设置的结构

struct Welcome: Decodable {
    let aapl, fb, msft, tsla, goog: Aapl

    enum CodingKeys: String, CodingKey {
        case aapl = "AAPL"
        case fb = "FB"
        case msft = "MSFT"
        case tsla = "TSLA"
        case goog = "GOOG"
    }
}

struct Aapl: Decodable {
    let quote: Quote
    let news: [News]
}

struct Quote: Decodable {
    let symbol: String
    let companyName: String
    let latestPrice: Double
}

struct News: Decodable {
    let url: String
    let image: String
}

这是获取数据的方式:
fetchData(url: stockApiUrl) { (result: FetchResult<Welcome>) -> (Void) in
    switch result {
    case .success(let object): self.stockData = [object]
        print("stockData: \n\n\(self.stockData)")
    case .failure(let error):
        print("Error decoding JSON: \n\n\(error)")
    }
    DispatchQueue.main.async {
        self.tableView.reloadData()
    }
}

并将数据传递到
var stockData = [Welcome]()
使用现在的代码,它可以输出公司所需的所有数据,但是我对于如何访问公司的所有数据感到困惑,因此我可以分别抓取每块(companyNamelatestPrice等),这样我就可以在表格视图中显示它们。

例如,在我的cellForRowAt表视图方法中,我试图将companyName设置为标签,但是使用let quotes = stockData[indexPath.row]会使我的索引超出范围,因此我不能再使用cell.companyNameLabel = quotes[indexPath.row].companyName之类的东西。

我想知道是否需要重组我的结构(尽管我将我的代码与quicktype.io的建议代码进行了比较,并且是相同的),还是重新进行了其他工作。

感谢您的指导!

编辑:
numberOfRows方法:
override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
    if section == 0 {
        return stockData.count
    } else if section == 1 && fetchingMore {
        return 1
    }
    return 0
}
fetchData函数:
func fetchData<T: Decodable>(url: URL, completion: @escaping (FetchResult<T>) -> (Void)) {

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

        guard let data = data else {completion(.failure(error!)); return}

        do {
            //let object = try JSONDecoder().decode(T.self, from: data)
            let object = try? JSONDecoder().decode([String:Aapl].self,from: data)
            completion(.success(object))
        } catch {
            completion(.failure(error))
        }
    }.resume()
}

最佳答案

要使用更通用和更具描述性的名称,请将Aapl重命名为Stock

struct Stock: Decodable {
    let quote: Quote
    let news: [News]
}

stockData声明为字典,并为键创建一个数组
var stockData = [String:Stock]()
var keys = [String]()

用替换fetchData
fetchData(url: stockApiUrl) { (result: FetchResult<[String:Stock]>) -> (Void) in
    switch result {
    case .success(let object):
        self.stockData = object
        self.keys = Array(object.keys)
        print("stockData: \n\n\(self.stockData)")
    case .failure(let error):
        print("Error decoding JSON: \n\n\(error)")
    }
    DispatchQueue.main.async {
        self.tableView.reloadData()
    }
}

cellForRow编写
let stock = stocks[keys[indexPath.row]]!
cell.companyNameLabel = stock.quote.companyName

10-08 07:43
查看更多