私は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)
}
}
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および手動パーセントエスケープレンディションについては、 この回答の以前のリビジョン を参照してください。
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/
私はこれを使用しています。遊び場で試してください。基本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
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)")
}
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
への偶発的な呼び出しを防ぎます
@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)!
}
}