将数据传递给segue.destination功能中



我正在尝试对SEGUE INDOKE进行2个API调用,并最终将数据数组从第二个调用传递到CollectionView。首次通话,我将获得一个值 catID ,我需要进行另一个呼叫:

let searchEndpoint: String = MY_ENDPOINT
// Add auth key
let serviceCallWithParams = searchEndpoint + "?PARAMETER"
guard let url = URL(string: serviceCallWithParams) else {
    print("Error: cannot create URL")
    return
}
let urlRequest = URLRequest(url: url)
// setting up the session
let config = URLSessionConfiguration.default
let session = URLSession(configuration: config)
// making the request
let task = session.dataTask(with: urlRequest) {
    (data, response, error) in
    // error check
    guard error == nil else {
        print("error")
        print(error)
        return
    }
    // make sure we got data
    guard let responseData = data else {
        print("Error: did not receive data")
        return
    }
    // parse JSON
    do {
        guard let catData = try JSONSerialization.jsonObject(with: responseData, options: []) as? [String: AnyObject] else {
            print("error converting data to JSON")
            return
        }
        if let data = catData["data"] as? [String: Any] {
            if let array = data["categories"] as? [Any] {
                if let firstObject = array.first as? [String: Any] {
                    if let catId = firstObject["catId"] as? Int {
                        getTitles(catId: catId)
                    }
                }
            }
        }
    } catch  {
        print("error converting data to JSON")
        return
    }
}
task.resume()

,然后getTitles函数看起来像这样:

func getTitles(catId: Int) {
    let catIdString = String(catId)
    let titlesEndPoint: String = MY_ENDPOINT + catIdString
    // Add auth key
    let titlesEndPointWithParams = titlesEndPoint + "?PARAMETER"
    guard let titlesUrl = URL(string: titlesEndPointWithParams) else {
        print("Error: cannot create URL")
        return
    }
    let titlesUrlRequest = URLRequest(url: titlesUrl)
    // set up the session
    let config = URLSessionConfiguration.default
    let session = URLSession(configuration: config)
    // make the request
    let task = session.dataTask(with: titlesUrlRequest) {
        (data, response, error) in
        // check for any errors
        guard error == nil else {
            print("error calling GET on listCategoryTitles")
            print(error)
            return
        }
        // make sure we got data
        guard let titlesData = data else {
            print("Error: did not receive data")
            return
        }
        // parse the JSON
        do {
            guard let allTitles = try JSONSerialization.jsonObject(with: titlesData, options: []) as? [String: AnyObject] else {
                print("error converting data to JSON")
                return
            }
            if let titlesJson = allTitles["data"] as? [String: Any] {
                if let titlesArray = titlesJson["titles"] as? Array<AnyObject> {
                    self.books = []
                    for (index, value) in titlesArray.enumerated() {
                        var book = Book()
                        book.bookTitle = value["title"] as? String
                        book.bookAuthor = value["author"] as? String
                        if let imageSource = value["_links"] as? Array<AnyObject> {
                            book.bookImageSource = imageSource[1]["href"] as? String
                        }
                        self.books?.append(book)
                    }
                }
            }
        } catch  {
            print("error converting data to JSON")
            return
        }
    }
    task.resume()
}

现在放置时:

let resultsVC = segue.destination as? CollectionViewController
resultsVC?.books = self.books

外部功能,在目标控制器中,我会在第一次单击时获得一个空数组作为输出,但是在下一个下,我将获得正确的数据。当我尝试将此内部函数放置时," getTitles" collectionViewController中的输出是" nil" 。值得一提的是,我有" books" so:

定义的变量

主控制器:

var books: [Book]? = []

收集控制器:

var books: [Book]?

我创建了类型[book],基本上是对象,在单独的结构中具有3个字符串变量。

上面的所有代码都封装在

override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
    if segue.identifier == "ShowResults" {

任何帮助/指南将不胜感激!

当您进行API调用时,它将在后台执行意味着asynchronously,在prepare(for:sender:)中将调用synchronously

现在,从您的问题来看,您已经在ButtonViewController中创建了故事板中的SEGUE,因此在您从API中获得响应之前,您已将其移至目标控制器,为了解决您的问题,您需要从您的问题中创建SEGUESource ViewController到Destination ViewController并设置其标识符。之后在getTitles(catId: Int)方法内进行循环后,在主线程上执行segue。

for (index, value) in titlesArray.enumerated() {
    var book = Book()
    book.bookTitle = value["title"] as? String
    book.bookAuthor = value["author"] as? String
    if let imageSource = value["_links"] as? Array<AnyObject> {
        book.bookImageSource = imageSource[1]["href"] as? String
    }
    self.books?.append(book)
}
DispatchQueue.main.async {
    self.performSegue(withIdentifier: "ShowResults", sender: nil)
}

之后,在您的prepare(for:sender:)内部进行更改。

override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
     if segue.identifier == "ShowResults" {
          let resultsVC = segue.destination as? CollectionViewController
          resultsVC?.books = self.books
     }
}

最新更新