Swift 4将数据从json保存到数组中以在TableView中显示它

时间:2017-12-30 10:23:24

标签: json swift uitableview tableview swift4

我正在尝试将数据从func getCoinData保存到数组sympolsCoin和数组sympolsCoin以在我的TableView中使用它

我在同一个ViewController.swift文件中创建了这个类:

struct Coin: Decodable {

let symbol : String
let price_usd : String } 

这在我的View控制器类中:

var coins = [Coin]()


var sympolsCoin = [String]()
var priceUSDcoin = [String]()



func getCoinData(completion: @escaping () -> ()) {
    let jsonURL = "https://api.coinmarketcap.com/v1/ticker/"
    let url = URL(string: jsonURL)

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

        do {
            self.coins = try JSONDecoder().decode([Coin].self, from: data!)

            for info in self.coins {

                self.sympolsCoin.append(info.symbol)
                self.priceUSDcoin.append(info.price_usd)

                print("\(self.sympolsCoin) : \(self.priceUSDcoin)")

                completion()
            }

        }


        catch {
            print("Error is : \n\(error)")
        }
    }.resume()
}

当我在TableView中使用数组时,我得到了空白表!

func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {

    let cell = tableView.dequeueReusableCell(withIdentifier: "BitcoinTableViewCell", for: indexPath) as! BitcoinTableViewCell

    cell.coinNameLable.text = sympolsCoin[indexPath.row]
    cell.priceLable.text = priceUSDcoin[indexPath.row]

    return cell

}

4 个答案:

答案 0 :(得分:3)

由于您使用的是JSONDecoder,因此创建和填充sympolsCoinpriceUSDcoin的整个逻辑毫无意义且冗余。

struct Coin: Decodable {
    private enum CodingKeys: String, CodingKey {
        case symbol, priceUSD = "price_usd"
    }
    let symbol : String
    let priceUSD : String 
} 
var coins = [Coin]()

完成处理程序也是多余的。只需在收到数据后重新加载主线程上的表视图:

func getCoinData() {
    let jsonURL = "https://api.coinmarketcap.com/v1/ticker/"
    let url = URL(string: jsonURL)

    URLSession.shared.dataTask(with: url!) { [unowned self] (data, response, error) in
        guard let data = data else { return }
        do {
            self.coins = try JSONDecoder().decode([Coin].self, from: data)
            DispatchQueue.main.async {
               self.tableView.reloadData()
            }

        } catch {
            print("Error is : \n\(error)")
        }
    }.resume()
}

viewDidLoad中加载数据

override func viewDidLoad() {
    super.viewDidLoad()
    getCoinData()
}

cellForRow更新用户界面

func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {

    let cell = tableView.dequeueReusableCell(withIdentifier: "BitcoinTableViewCell", for: indexPath) as! BitcoinTableViewCell

    let coin = coins[indexPath.row]
    cell.coinNameLable.text = coin.symbol
    cell.priceLable.text = coin.priceUSD

    return cell

}

答案 1 :(得分:2)

您需要从主线程更新tableView。作为一个值得学习的好教训:始终从主线程更新UI。总是

do {
        self.coins = try JSONDecoder().decode([Coin].self, from: data!)

        for info in self.coins {

            self.sympolsCoin.append(info.symbol)
            self.priceUSDcoin.append(info.price_usd) 

            DispatchQueue.main.async {
                self.tableView.reloadData()
            }

            print("\(self.sympolsCoin) : \(self.priceUSDcoin)")

            completion()
        }

    }

但是,您的代码的另一个问题是您的标签设置方式无效。 TableViewCells被重用了,所以我猜你在其他地方有@IBOutlets。你应该做的是在cellForRowAt中声明一个标签常量:

override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
    let cell = tableView.dequeueReusableCell(withIdentifier: "reuseIdentifier", for: indexPath)

    let coinNameLabel = cell.viewWithTag(100) as! UILabel
    coinNameLabel.text = sympolsCoin[indexPath.row]
    let priceNameLabel = cell.viewWithTag(101) as! UILabel
    priceNameLabel.text = priceUSDcoin[indexPath.row]
}

上面的代码假设您在故事板中设置了标签100和101的两个标签(假设您使用了标签)

答案 2 :(得分:1)

在ViewController类中创建一个tableView插座,然后将其命名为“tableView” 试试这段代码:Swift 4

func getCoinData() {
    let jsonURL = "https://api.coinmarketcap.com/v1/ticker/"
    let url = URL(string: jsonURL)

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

        do {
            self.coins = try JSONDecoder().decode([Coin].self, from: data!)

            for info in self.coins {

                self.sympolsCoin.append(info.symbol)
                self.priceUSDcoin.append(info.price_usd)

                print("\(self.sympolsCoin) : \(self.priceUSDcoin)")

                self.tableView.reloadData()
            }

        }

        catch {
            print("Error is : \n\(error)")
        }
  }.resume()
}

在ViewDidLoad中调用此函数,如下所示

 override func viewDidLoad() {
    super.viewDidLoad()
    getCoinData()
}

答案 3 :(得分:-1)

**

// First View Controller
//
//
//
import UIKit
struct Countory : Decodable {
    let name: String
    let capital: String
    let region: String
    let alpha2Code: String
}
class ViewController: UIViewController, UITableViewDelegate, UITableViewDataSource {
  var listArr = [Countory]()
    @IBOutlet weak var tableView: UITableView! 
    override func viewDidLoad() {
        super.viewDidLoad()
        self.tableView.delegate = self
        self.tableView.dataSource = self
        let url = "https://restcountries.eu/rest/v2/all"
        let urlObj = URL(string: url)!
        URLSession.shared.dataTask(with: urlObj) {(data, responds, Error) in
         do {
          self.listArr = try JSONDecoder().decode([Countory].self, from: data!)
             for country in self.listArr {

                    print("Country",country.name)
                    print("###################")
                    print("Capital",country.capital)
                 DispatchQueue.main.async {
                       self.tableView.reloadData()
                    }
                }
            } catch {
                print(" not ")
            }
        }.resume()
    }
     func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return self.listArr.count
    }

func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "TableViewCell") as! TableViewCell
        cell.label1.text = "Name: \(listArr[indexPath.row].name)"
        cell.lable2.text = listArr[indexPath.row].capital
        return cell
    }

    func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
        let homeView = self.storyboard?.instantiateViewController(withIdentifier: "SecondViewController") as! SecondViewController
          homeView.res = listArr[indexPath.row].region
          homeView.alpha = listArr[indexPath.row].alpha2Code
       self.navigationController?.pushViewController(homeView, animated: true)

    }

    func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {
        return 100
    }
}
// SecondViewController
class SecondViewController: UIViewController {
    @IBOutlet weak var label4: UILabel!
    @IBOutlet weak var label3: UILabel!

    var res = ""
    var alpha = ""

    override func viewDidLoad() {
        super.viewDidLoad()

        self.label3.text = res
        self.label4.text = alpha

    }

}

**