我正在尝试从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]()
使用现在的代码,它可以输出公司所需的所有数据,但是我对于如何访问公司的所有数据感到困惑,因此我可以分别抓取每块(
companyName
,latestPrice
等),这样我就可以在表格视图中显示它们。例如,在我的
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