Code example: https://github.com/johnlui/Swift-On-iOS/blob/master/BuildYourHTTPRequestLibrary Open source project: Pitaya, an HTTP request library suitable for uploading large files: https://github.com/johnlui/Pitaya In this article, we will try to use a class to encapsulate our previous code, try to add the function of dynamically adding HTTP parameters (params), and then encapsulate a powerful interface. [[135523]] Basic package Basic preparation Create a new empty Swift file, name it Network.swift, write a Network class in it, and then write a static method request(): - class Network{
- static func request() {
- let session = NSURLSession .sharedSession()
- let request = NSURLRequest (URL: NSURL(string: "http://baidu.com")!)
- let task = session .dataTaskWithRequest(request, completionHandler: { (data, response, error) - > Void in
- println("just wait for 5 seconds!")
- sleep(5)
- let string = NSString (data: data, encoding: NSUTF8StringEncoding)
- println(string)
- })
- task.resume()
- }
- }
Modify the button function in ViewController: - @IBAction func mainButtonBeTapped(sender: AnyObject) {
- Network.request()
- }
Run the project and click the button. The effect is the same as before. Custom HTTP method and URL Modify the request() method and pass in the HTTP method and URL: - static func request(method: String, url: String) {
- let session = NSURLSession.sharedSession()
- let request = NSMutableURLRequest(URL: NSURL(string: url)!)
- request.HTTPMethod = method
- let task = session.dataTaskWithRequest(request, completionHandler: { (data, response, error) -> Void in
- println( "just wait for 5 seconds!" )
- sleep( 5 )
- let string = NSString(data: data, encoding: NSUTF8StringEncoding)
- println(string)
- })
- task.resume()
- }
Modify the previous function call: - @IBAction func mainButtonBeTapped(sender: AnyObject) {
- Network.request( "GET" , url: "http://baidu.com" )
- }
Run the project and click the button. The effect is the same as before. Use closures to handle request results Functions are first-class citizens in Swift. Closures can be used as function parameters and return values, which is very powerful. Next, we use closures to process the return value of network requests. Modify the request() method and pass in a closure: - static func request(method: String, url: String, callback: (data: NSData!, response: NSURLResponse!, error: NSError!) -> Void) {
- let session = NSURLSession.sharedSession()
- let request = NSMutableURLRequest(URL: NSURL(string: url)!)
- request.HTTPMethod = method
- let task = session.dataTaskWithRequest(request, completionHandler: { (data, response, error) -> Void in
- callback(data: data, response: response, error: error)
- })
- task.resume()
- }
Use closures to process results in the previous function call:
- @IBAction func mainButtonBeTapped(sender: AnyObject) {
- Network.request( "GET" , url: "http://baidu.com" ) { (data, response, error) -> Void in
- println( "just wait for 5 seconds!" )
- sleep( 5 )
- let string = NSString(data: data, encoding: NSUTF8StringEncoding)
- println(string)
- }
- }
Run the project and click the button. The effect is the same as before. Dynamically add Params GET method Under the GET method, params is directly attached to the end of the URL after being url-encoded and sent to the server. Modify the request() method and pass in a params dictionary: - static func request(method: String, url: String, params: Dictionary = Dictionary(), callback: (data: NSData!, response: NSURLResponse!, error: NSError!) -> Void) {
- ... ...
- }
To process params, we steal the params processing function from Alamofire. If it is a GET method, then add the processed params to the end of the URL. The complete code of the Network class is as follows:
- class Network{
- static func request(method: String, url: String, params: Dictionary = Dictionary(), callback: (data: NSData!, response: NSURLResponse!, error: NSError!) -> Void) {
- let session = NSURLSession.sharedSession()
-
- var newURL = url
- if method == "GET" {
- newURL += "?" + Network().buildParams(params)
- }
-
- let request = NSMutableURLRequest(URL: NSURL(string: newURL)!)
- request.HTTPMethod = method
-
- let task = session.dataTaskWithRequest(request, completionHandler: { (data, response, error) -> Void in
- callback(data: data, response: response, error: error)
- })
- task.resume()
- }
-
-
- func buildParams(parameters: [String: AnyObject]) -> String {
- var components: [(String, String)] = []
- for key in sorted(Array(parameters.keys), [(String, String)] {
- var components: [(String, String)] = []
- if let dictionary = value as? [String: AnyObject] {
- for (nestedKey, value) in dictionary {
- components += queryComponents( "\(key)[\(nestedKey)]" , value)
- }
- } else if let array = value as? [AnyObject] {
- for value in array {
- components += queryComponents( "\(key)" , value)
- }
- } else {
- components.extend([(escape(key), escape( "\(value)" ))])
- }
-
- return components
- }
- func escape(string: String) -> String {
- let legalURLCharactersToBeEscaped: CFStringRef = ":&=;+!@#$()',*"
- return CFURLCreateStringByAddingPercentEscapes(nil, string, nil, legalURLCharactersToBeEscaped, CFStringBuiltInEncodings.UTF8.rawValue) as String
- }
- }
Modify the previous function call: - @IBAction func mainButtonBeTapped(sender: AnyObject) {
- Network.request( "GET" , url: "http://pitayaswift.sinaapp.com/pitaya.php" , params: [ "get" : "Network" ]) { (data, response, error) -> Void in
- let string = NSString(data: data, encoding: NSUTF8StringEncoding)
- println(string)
- }
- }
http://pitayaswift.sinaapp.com/pitaya.php is the server code I deployed for testing, which will directly return ooxx in ?get=ooxx. Run the project and click the button to see the effect: POST method There are several protocols to choose from under the POST method. There is no file upload here, so we use the simpler application/x-www-form-urlencoded method to send the request. Add some code to the request() method: - static func request(method: String, url: String, params: Dictionary = Dictionary(), callback: (data: NSData!, response: NSURLResponse!, error: NSError!) -> Void) {
- let session = NSURLSession.sharedSession()
-
- var newURL = url
- if method == "GET" {
- newURL += "?" + Network().buildParams(params)
- }
-
- let request = NSMutableURLRequest(URL: NSURL(string: newURL)!)
- request.HTTPMethod = method
-
- if method == "POST" {
- request.addValue( "application/x-www-form-urlencoded" , forHTTPHeaderField: "Content-Type" )
- request.HTTPBody = Network().buildParams(params).dataUsingEncoding(NSUTF8StringEncoding)
- }
-
- let task = session.dataTaskWithRequest(request, completionHandler: { (data, response, error) -> Void in
- callback(data: data, response: response, error: error)
- })
- task.resume()
- }
Modify the previous function call: - @IBAction func mainButtonBeTapped(sender: AnyObject) {
- Network.request( "POST" , url: "http://pitayaswift.sinaapp.com/pitaya.php" , params: [ "post" : "Network" ]) { (data, response, error) -> Void in
- let string = NSString(data: data, encoding: NSUTF8StringEncoding)
- println(string)
- }
- }
Use POST to send a request, and the server will also return the value of the key post. Run the project and click the button. The result is the same as the previous GET method. At this point, the interface encapsulation is completed! |