web-dev-qa-db-ja.com

Swiftパラメータ付きのGETリクエスト

私はSwiftが初めてなので、おそらくコードに多くの障害がありますが、私が達成しようとしているのは、パラメーターを使用してGET要求をlocalhostサーバーに送信することです。さらに、関数が2つのパラメータbaseURL:string,params:NSDictionaryを取得することを実現しようとしています。これら2つを実際のURLRequestにどのように組み合わせるのかわかりません。ここに私が今まで試したものがあります

    func sendRequest(url:String,params:NSDictionary){
       let urls: NSURL! = NSURL(string:url)
       var request = NSMutableURLRequest(URL:urls)
       request.HTTPMethod = "GET"
       var data:NSData! =  NSKeyedArchiver.archivedDataWithRootObject(params)
       request.HTTPBody = data
       println(request)
       var session = NSURLSession.sharedSession()
       var task = session.dataTaskWithRequest(request, completionHandler:loadedData)
       task.resume()

    }

}

func loadedData(data:NSData!,response:NSURLResponse!,err:NSError!){
    if(err != nil){
        println(err?.description)
    }else{
        var jsonResult: NSDictionary = NSJSONSerialization.JSONObjectWithData(data, options: NSJSONReadingOptions.MutableContainers, error: nil) as NSDictionary
        println(jsonResult)

    }

}
64
MrSSS16

GETリクエストを作成する場合、リクエストの本文はありませんが、すべてがURLに送信されます。 URLを構築する(および適切にエスケープする)には、URLComponentsを使用することもできます。

var url = URLComponents(string: "https://www.google.com/search/")!

url.queryItems = [
    URLQueryItem(name: "q", value: "War & Peace")
]

唯一のトリックは、ほとんどのWebサービスで+文字パーセントをエスケープする必要があることです( application/x-www-form-urlencoded仕様 で指定されているように、スペース文字として解釈されるため)。ただし、URLComponentsはパーセントエスケープされません。 Appleは、+がクエリ内の有効な文字であるため、エスケープしないでくださいと主張します。技術的には、URIのクエリで許可されていますが、application/x-www-form-urlencodedリクエストでは特別な意味があり、エスケープせずに渡すことはできません。

Appleは、+文字をパーセントでエスケープする必要があることを認めていますが、手動で行うことをお勧めします。

var url = URLComponents(string: "https://www.wolframalpha.com/input/")!

url.queryItems = [
    URLQueryItem(name: "i", value: "1+2")
]

url.percentEncodedQuery = url.percentEncodedQuery?.replacingOccurrences(of: "+", with: "%2B")

これは洗練されていない回避策ですが、機能します。また、クエリに+文字が含まれていて、それらをスペースとして解釈するサーバーがある場合は、Appleがアドバイスします。

したがって、それをsendRequestルーチンと組み合わせると、次のような結果になります。

func sendRequest(_ url: String, parameters: [String: String], completion: @escaping ([String: Any]?, Error?) -> Void) {
    var components = URLComponents(string: url)!
    components.queryItems = parameters.map { (key, value) in 
        URLQueryItem(name: key, value: value) 
    }
    components.percentEncodedQuery = components.percentEncodedQuery?.replacingOccurrences(of: "+", with: "%2B")
    let request = URLRequest(url: components.url!)

    let task = URLSession.shared.dataTask(with: request) { data, response, error in
        guard let data = data,                            // is there data
            let response = response as? HTTPURLResponse,  // is there HTTP response
            (200 ..< 300) ~= response.statusCode,         // is statusCode 2XX
            error == nil else {                           // was there no error, otherwise ...
                completion(nil, error)
                return
        }

        let responseObject = (try? JSONSerialization.jsonObject(with: data)) as? [String: Any]
        completion(responseObject, nil)
    }
    task.resume()
}

そして、あなたはそれを次のように呼ぶでしょう:

sendRequest("someurl", parameters: ["foo": "bar"]) { responseObject, error in
    guard let responseObject = responseObject, error == nil else {
        print(error ?? "Unknown error")
        return
    }

    // use `responseObject` here
}

個人的には、最近ではJSONDecoderを使用し、辞書ではなくカスタムstructを返しますが、ここではあまり関係ありません。うまくいけば、これがパラメータをGETリクエストのURLにパーセントエンコードする方法の基本的な考え方を示しています。


Swift 2および手動パーセントエスケープレンディションについては、 この回答の以前のリビジョン を参照してください。

115
Rob

NSURLComponentsを使用して、このようにNSURLを構築します

var urlComponents = NSURLComponents(string: "https://www.google.de/maps/")!

urlComponents.queryItems = [
  NSURLQueryItem(name: "q", value: String(51.500833)+","+String(-0.141944)),
  NSURLQueryItem(name: "z", value: String(6))
]
urlComponents.URL // returns https://www.google.de/maps/?q=51.500833,-0.141944&z=6

フォント: https://www.ralfebert.de/snippets/ios/encoding-nsurl-get-parameters/

86
Ben-Hur Batista

私はこれを使用しています。遊び場で試してください。基本URLを定数の構造体として定義する

struct Constants {

    struct APIDetails {
        static let APIScheme = "https"
        static let APIHost = "restcountries.eu"
        static let APIPath = "/rest/v1/alpha/"
    }
}

private func createURLFromParameters(parameters: [String:Any], pathparam: String?) -> URL {

    var components = URLComponents()
    components.scheme = Constants.APIDetails.APIScheme
    components.Host   = Constants.APIDetails.APIHost
    components.path   = Constants.APIDetails.APIPath
    if let paramPath = pathparam {
        components.path = Constants.APIDetails.APIPath + "\(paramPath)"
    }
    if !parameters.isEmpty {
        components.queryItems = [URLQueryItem]()
        for (key, value) in parameters {
            let queryItem = URLQueryItem(name: key, value: "\(value)")
            components.queryItems!.append(queryItem)
        }
    }

    return components.url!
}

let url = createURLFromParameters(parameters: ["fullText" : "true"], pathparam: "IN")

//Result url= https://restcountries.eu/rest/v1/alpha/IN?fullText=true
4
anoop4real

Swift

extension URL {
    func getQueryItemValueForKey(key: String) -> String? {
        guard let components = NSURLComponents(url: self, resolvingAgainstBaseURL: false) else {
              return nil
        }

        guard let queryItems = components.queryItems else { return nil }
     return queryItems.filter {
                 $0.name.lowercased() == key.lowercased()
                 }.first?.value
    }
}

func imagePickerController(_ picker: UIImagePickerController, didFinishPickingMediaWithInfo info: [String : Any])UIImagePickerControllerの画像名を取得するために使用しました:

var originalFilename = ""
if let url = info[UIImagePickerControllerReferenceURL] as? URL, let imageIdentifier = url.getQueryItemValueForKey(key: "id") {
    originalFilename = imageIdentifier + ".png"
    print("file name : \(originalFilename)")
}
2
Danut Pralea

Dictionaryを拡張して、キーと値の両方がstringFromHttpParameterに適合する場合にのみCustomStringConvertableを提供できます。

extension Dictionary where Key : CustomStringConvertible, Value : CustomStringConvertible {
  func stringFromHttpParameters() -> String {
    var parametersString = ""
    for (key, value) in self {
      parametersString += key.description + "=" + value.description + "&"
    }
    return parametersString
  }
}

これははるかにクリーンで、そのメソッドを呼び出すビジネスがない辞書でのstringFromHttpParametersへの偶発的な呼び出しを防ぎます

0
Reza Shirazian

@Robが提案したこの拡張機能は、Swift 3.0.1で機能します

Xcode 8.1(8B62)で彼の投稿に含まれているバージョンをコンパイルできませんでした

extension Dictionary {

    /// Build string representation of HTTP parameter dictionary of keys and objects
    ///
    /// :returns: String representation in the form of key1=value1&key2=value2 where the keys and values are percent escaped

    func stringFromHttpParameters() -> String {

        var parametersString = ""
        for (key, value) in self {
            if let key = key as? String,
               let value = value as? String {
                parametersString = parametersString + key + "=" + value + "&"
            }
        }
        parametersString = parametersString.substring(to: parametersString.index(before: parametersString.endIndex))
        return parametersString.addingPercentEncoding(withAllowedCharacters: .urlHostAllowed)!
    }

}
0
etayluz