0

JSONを使用してデータを読み取り、コールバックを提供するネットワーク接続があります。

executeRequestURL(requestURL: url, taskCallback: {(status, resp) -> Void in

            if (status == true) {
                if let results = resp as? NSDictionary {
                    print ("\(results.count) results found")
                    let list = results.allValues.first as! NSArray

                    print (list)
                }

            } else {
                print ("Error -- \(resp)")
            }
        })

これは呼び出します。

private class func executeRequestURL(requestURL: NSURL, taskCallback: @escaping (Bool, AnyObject?) -> ()) {
        print ("Attempting URL -- \(requestURL)")

        let request: NSURLRequest = NSURLRequest(url: requestURL as URL, cachePolicy: .reloadIgnoringLocalAndRemoteCacheData, timeoutInterval: kAPI_TIMEOUT)

        let session: URLSession = URLSession.shared


        let task = session.dataTask(with: request as URLRequest, completionHandler: {
            (data, response, error) in

            guard error == nil else {
                print(error)
                return
            }
            guard let data = data else {
                print("Data is empty")
                return
            }

            let json = try! JSONSerialization.jsonObject(with: data, options: [])
            //print(json)

            if let response = response as? HTTPURLResponse , 200...299 ~= response.statusCode {
                taskCallback(true, json as AnyObject?)
            } else {
                taskCallback(false, json as AnyObject?)
            }

        })
        task.resume()
    }

私が抱えている問題は、結果を辞書に読み込み、それをループしてオブジェクトを作成したいということです。

今のところ、コードが確実に機能するように executeRequestURL にコードを配置しますが、必要なエンティティ用にこのコードを分離するつもりです。

質問:

respを辞書として読むにはどうすればよいですか?

ありがとう

サンプル応答は次のとおりです。

{
  "objects": [
    {
      "uid": "coll_20ce39424470457c925f823fc150b3d4",
      "title": "Popular",
      "temp_image": "",
      "body": "",
      "active": true,
      "slug": "popular",
      "created": "2014-10-25T12:45:54+00:00",
      "modified": "2014-10-25T12:45:54.159000+00:00",
      "ends_on": "2100-01-01T00:00:00+00:00",
    }
  ]
}
4

3 に答える 3

2

[String:Any]JSON は辞書なので、コールバックから辞書 ( ) を返します。Swift 3AnyObjectではAny. Swift の強力な型システムは、常に可能な限り具体的にすることを奨励しています。

より良いエラー処理を行います! だけではなく、エラーを返す必要がありますfalse

コードは新しい Swift 3 構造体を使用しURLURLRequest

private class func executeRequestURL(requestURL: URL, taskCallback: @escaping (Bool, [String:Any]?) -> ()) {
  print ("Attempting URL -- \(requestURL)")

  let request = URLRequest(url: requestURL, cachePolicy: .reloadIgnoringLocalAndRemoteCacheData, timeoutInterval: kAPI_TIMEOUT)

  let session = URLSession.shared


  let task = session.dataTask(with: request, completionHandler: {
    (data, response, error) in

    guard error == nil else {
      print(error)
      taskCallback(false, nil)
      return
    }
    guard let data = data else {
      print("Data is empty")  // <- this will never be reached. If there is no error,
      taskCallback(false, nil) // data is always non-nil.
      return
    }
    if let response = response as? HTTPURLResponse , 200...299 ~= response.statusCode {
      let json = try! JSONSerialization.jsonObject(with: data, options: []) as!  [String:Any]
      taskCallback(true, json)
    } else {
      taskCallback(false, nil)
    }
  })
  task.resume()
}

JSON の結果には、辞書objectsの配列を含む 1 つのキーを持つ辞書が含まれます。JSON コレクション型は非常に簡単に区別できます:{}は辞書、[]は配列です。


JSON をオブジェクトにマップするには、構造体を作成します

struct Item {

  var uid : String
  var title : String
  var tempImage : String
  var body : String
  var active : Bool
  var slug : String
  var created : String
  var modified : String
  var endOn : String
}

と配列

var items = [Item]()

次に、辞書をにマップしますItem

if let objects = json["objects"] as? [[String:Any]] {
  for object in objects {
    let uid = object["uid"] as! String
    var title = object["title"] as! String
    var tempImage = object["temp_image"] as! String
    var body = object["body"] as! String
    var active = object["active"] as! Bool
    var slug = object["slug"] as! String
    var created = object["created"] as! String
    var modified  = object["modified"] as! String
    var endOn  = object["end_on"] as! String
    let item = Item(uid: uid, title: title, tempImage:tempImage, body: body, active: active, slug: slug, created: created, modified: modified, endOn: endOn)
    items.append(item)

  }

JSON 値は、常にすべてのフィールドを含むデータベースから取得されているように見えるため、強制的にラップ解除された値は安全です。

于 2016-11-01T11:47:06.667 に答える
1

他にも多くの方法がありますが、私はObjectMapperを使用して行うのが好きです。私にはきれいに見えます。新しい Swift ファイルを作成し、インポートObjectMapperして以下のコードを記述します。

 class yourDataModel: Mappable {

// MARK: - Constants & Variables

var myObjects: [yourDataModel]

required init?(_ map: Map) {
    myObjects = []

}

func mapping(map: Map) {
    myObjects               <- map["objects"]
}

}

class YourCustomObjects: Mappable {

// MARK: - Constants & Variables

var userId:String
var title:String
var tempimage:String
var body:String
var active:Bool
var slug : String
var createdDate:String
var modifiedDate:String
var endDate:String

// MARK: - init

required init?(_ map: Map) {

    userId = ""
    title = ""
    tempimage = ""
    body = ""
    active = false
    slug = ""
    createdDate = ""
    modifiedDate = ""
    endDate = ""

}

func mapping(map: Map) {

    userId                      <- map["uid"]
    title                       <- map["title"]
    tempimage                   <- map["temp_image"]
    body                        <- map["body"]
    active                      <- map["active"]
    slug                        <- map["slug"]
    createdDate                 <- map["created"]
    modifiedDate                <- map["modified"]
    endDate                     <- map["ends_on"]

}

}

基本的にはモデルクラスです。結果をJSONで渡すだけで、うまくいけばAnyObjectになり、すべての「オブジェクト」を含む配列が得られます。以下のように使えます

if let data = Mapper<yourDataModel>().map(resp){
    print(data)
 }

これを試してみて、問題が発生した場合はお知らせください。

于 2016-11-01T11:35:53.987 に答える
1

私はそれを次のようにしました:

func getHttpData(urlAddress : String)
    {
        // Asynchronous Http call to your api url, using NSURLSession:
        guard let url = URL(string: urlAddress) else
        {
            print("Url conversion issue.")
            return
        }
        URLSession.shared.dataTask(with: url, completionHandler: { (data, response, error) -> Void in
            // Check if data was received successfully
            if error == nil && data != nil {
                do {
                    // Convert NSData to Dictionary where keys are of type String, and values are of any type
                    let json = try JSONSerialization.jsonObject(with: data!, options: JSONSerialization.ReadingOptions.mutableContainers) as! [String:AnyObject]
                    // Call whatever function you want to do with your dictionary
                    useMyDictionary(dictionary: json)
                } catch {
                    print(error)
                    // Something went wrong
                }
            }
            else if error != nil
            {
                print(error)
            }
        }).resume()
    }
于 2016-11-01T11:17:27.990 に答える