質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

90.49%

  • iOS

    4766questions

    iOSとは、Apple製のスマートフォンであるiPhoneやタブレット端末のiPadに搭載しているオペレーションシステム(OS)です。その他にもiPod touch・Apple TVにも搭載されています。

  • Swift 2

    1340questions

    Swift 2は、Apple社が独自に開発を行っている言語「Swift」のアップグレード版です。iOSやOS X、さらにLinuxにも対応可能です。また、throws-catchベースのエラーハンドリングが追加されています。

Alamofire + SwiftyJSONでreturnがnilになる

受付中

回答 1

投稿

  • 評価
  • クリップ 0
  • VIEW 1,348

hayaken

score 2

前提・実現したいこと

Alamofire+SwiftyJSONを利用して、APIからデータを取得・パースして、returnで戻したい。

発生している問題・エラーメッセージ

return値が変数を初期化した際の物([])が返却される。
returnされた後にAlamofireの処理を行っている模様(以下のログを参照)

該当のソースコード

 APIの処理元

import Foundation
import Alamofire
import SwiftyJSON

class CardApi {
  let email = AppSetting.email
  let token = AppSetting.token
  let host = Constants.Api.host
  var cards: [[String: String?]] = []

  func getCardList() -> [[String: String?]] {
    let url = host + "/card"
    let params = [
      "email" : email,
      "token" : token
    ]

    Alamofire.request(.GET, url, parameters: params)
      .responseJSON{ response in
        guard let object = response.result.value else {
          return
        }

        let json = JSON(object)
        json.forEach { (_, json) in
          let card: [String: String?] = [
            "id" : json["id"].description,
            "description" : json["description"].string
          ]
          self.cards.append(card)
        }
    }
    return cards
  }
}

 APIクラスの呼び出し元

import UIKit

class HomeViewController: UIViewController {

  @IBOutlet weak var userEmail: UILabel!
  @IBOutlet weak var userToken: UILabel!
  let cardApi = CardApi()

  override func viewDidLoad() {
    super.viewDidLoad()
    userEmail.text = AppSetting.email
    userToken.text = AppSetting.token

    print(NSDate())
    print("呼んだよ")
    print(cardApi.getCardList())
    print(NSDate())
    print("終わり")
    // Do any additional setup after loading the view.
  }

  override func didReceiveMemoryWarning(){
    super.didReceiveMemoryWarning()
    // Dispose of any resources that can be recreated.
  }
}

試したログ

2016-08-30 23:33:25 +0000
呼んだよ
[]
2016-08-30 23:33:25 +0000
終わり
[["id": Optional("1"), "description": Optional("あああああああああ")]]
[["id": Optional("1"), "description": Optional("あああああああああ")], ["id": Optional("2"), "description": Optional("いいいいいいいいいいい")]]

補足情報(言語/FW/ツール等のバージョンなど)

より詳細な情報

  • Xcode: 7.3.1
  • Swift: 2.2
  • Alamofire: 3.4.2
  • SwiftyJSON: 2.3.2
  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

質問への追記・修正、ベストアンサー選択の依頼

  • 退会済みユーザー

    2016/08/31 10:52

    こちらの質問が他のユーザから「やってほしいことだけを記載した丸投げの質問」という指摘を受けました
    「質問を編集する」ボタンから編集を行い、調査したこと・試したことを記入していただくと、回答が得られやすくなります。

回答 1

0

実際には試せてませんが以下のようにクロージャーを使用して返却するのが良いと思います。

import Foundation
import Alamofire
import SwiftyJSON

class CardApi {

    let email = AppSetting.email
    let token = AppSetting.token
    let host = Constants.Api.host
    var cards: [[String: String?]] = []

    func getCardList(resultClosure: [[String: String?]] -> Void) {

        let url = host + "/card"
        let params = [
            "email" : email,
            "token" : token
        ]

        Alamofire.request(.GET, url, parameters: params)
            .responseJSON{ response in
                guard let object = response.result.value else {
                    return
                }

                let json = JSON(object)
                json.forEach { (_, json) in
                    let card: [String: String?] = [
                        "id" : json["id"].description,
                        "description" : json["description"].string
                    ]
                    self.cards.append(card)
                }

                resultClosure(self.cards)
        }
    }
}


class HomeViewController: UIViewController {

    @IBOutlet weak var userEmail: UILabel!
    @IBOutlet weak var userToken: UILabel!
    let cardApi = CardApi()

    override func viewDidLoad() {
        super.viewDidLoad()
        userEmail.text = AppSetting.email
        userToken.text = AppSetting.token

        print(NSDate())
        print("呼んだよ")
        cardApi.getCardList { data in
            print(data)
        }
        print(NSDate())
        print("終わり")
        // Do any additional setup after loading the view.
    }

    override func didReceiveMemoryWarning(){
        super.didReceiveMemoryWarning()
        // Dispose of any resources that can be recreated.
    }
}

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2016/08/31 10:40

    回答有り難うございます!
    ただ、試してみたのですが、クロージャーからの渡ってきた dataがprintしてもなにも表示されません(´・ω・`)

    ```
    2016-08-31 01:38:32 +0000
    呼んだよ
    2016-08-31 01:38:32 +0000
    終わり
    ```

    キャンセル

  • 2016/08/31 12:16 編集

    ちなみにdataのブロックの中は呼ばれていますか?

    あとresultClosure(self.cards)を呼び出す前に
    print(self.cards)を記述すると値は出力されますか?

    キャンセル

同じタグがついた質問を見る

  • iOS

    4766questions

    iOSとは、Apple製のスマートフォンであるiPhoneやタブレット端末のiPadに搭載しているオペレーションシステム(OS)です。その他にもiPod touch・Apple TVにも搭載されています。

  • Swift 2

    1340questions

    Swift 2は、Apple社が独自に開発を行っている言語「Swift」のアップグレード版です。iOSやOS X、さらにLinuxにも対応可能です。また、throws-catchベースのエラーハンドリングが追加されています。