因为iOS的权限限制, 如果使用HTTP协议要配置info.plist, 将Allow Arbitary Loads设为YES。
iOS封装了URLSession类处理HTTP交互, 支持交互文本、上传文件、下载文件。
一、 文本交互
一般是用POST请求将包体数据传给后台, 后台返回json包体给手机端, 手机端解析json后做逻辑。
<pre> let urlStr = "http://baike.baidu.com/api/openapi/BaikeLemmaCardApi?scope=103&format=json&appid=379020&bk_key=swift&bk_length=600"
let url = URL(string: urlStr)
var request = URLRequest(url: url!) //请求
request.httpMethod = "POST" //修改http方法
//request.httpBody = Data(bytes: <#T##Array<UInt8>#>) //设置POST包体
let session = URLSession.shared
let date = Date()
print("创建任务, 时间:\(date.timeIntervalSince1970)")
//初始化请求
let dataTask = session.dataTask(with: request,
completionHandler: { (data, resp, err) in
let comDate = Date()
print("http返回, 时间:\(comDate.timeIntervalSince1970)")
if err != nil {
print(err.debugDescription)
} else {
let responseStr = String(data: data!,
encoding: String.Encoding.utf8)
//print(responseStr!) //包体数据
//print("mimeType: (resp?.mimeType) ")
//URLResponse类里没有http返回值, 需要先强制转换!
if let response = resp as? HTTPURLResponse {
print("code\ (response.statusCode)")
for (tab, result) in response.allHeaderFields {
print("(tab.description) - (result)")
}
if response.statusCode == 200 {
//JSON解析, 做逻辑
} else {
//通知UI接口执行失败
}
}
}
} ) as URLSessionTask
let beginDate = Date()
print("开始任务, 时间:\(beginDate.timeIntervalSince1970)")
dataTask.resume() //执行任务
let endDate = Date()
print("结束任务, 时间:\(endDate.timeIntervalSince1970))</pre>
这段代码说明几个问题:
1、 request.httpMethod参数可以修改HTTP的方法, 默认是GET。
2、dataTask.resume()是异步执行的,即不阻塞UI。 这里还有闭包的一个概念叫逃逸闭包,对应关键字@escapting, 它的意思是将闭包做为回调异步执行(作用类似于Android的Runnable),调用时立刻返回。
<pre> open func dataTask(with request: URLRequest, completionHandler: @escaping (Data?, URLResponse?, Error?) -> Swift.Void) -> URLSessionDataTask</pre>执行日志:
<pre>创建任务, 时间:1484230880.29283
开始任务, 时间:1484230880.30001
结束任务, 时间:1484230880.30009
http返回, 时间:1484230881.42724 </pre>
3、 http交互成功后要判断返回值, 作为初学者我翻遍了URLResponse的方法, 就是没有status code。。。 后来无意中发现了HTTPResponse类, 试着强转并输出它的成员变量, 果然好用。
<pre>code 200
Server - Apache
Content-Type - application/json
Transfer-Encoding - Identity
Date - Thu, 12 Jan 2017 14:21:21 GMT
Proxy-Connection - Keep-alive
Tracecode - 12812437700874983946011222</pre>
4、http执行成功后就是要解析包体并做业务逻辑了, responseStr就是我们最终需要的json字符串, 我们需要反序列化并做逻辑。
<pre> if response.statusCode == 200 {
//JSON解析, 做逻辑
} else {
//通知UI接口执行失败
}</pre>
二、 下载文件, 使用URLSession的API, 代码很简单。 重点是存储位置, iOS会自动生成一个临时文件。 我们要做的是拷贝这个文件到我们想要的目录下。
<pre>let url = URL(string: "http://img.pconline.com.cn/images/upload/upc/tx/wallpaper/1307/23/c0/23656308_1374564438338_800x600.jpg")
let request = URLRequest(url: url!)
let downloadTask = session.downloadTask(with: request)
downloadTask.resume() //开始下载任务
func urlSession(_ session: URLSession, downloadTask: URLSessionDownloadTask, didFinishDownloadingTo location: URL) {
print("下载结束, 存储在(location.path)")
}
func urlSession(_ session: URLSession, downloadTask: URLSessionDownloadTask, didWriteData bytesWritten: Int64, totalBytesWritten: Int64, totalBytesExpectedToWrite: Int64) {
print("total: \(totalBytesWritten), current: \(bytesWritten)") //下载进度
}
</pre>
日志:
<pre> total: 46276, current: 46276
下载结束, 存储在/Users/brycegao/Library/Developer/CoreSimulator/Devices/8BE9C62E-042E-4B50-8F5D-78F857533650/data/Containers/Data/Application/3A6EB44A-2F88-4E79-9CFC-87713B9FC2E0/tmp/CFNetworkDownload_tnAj6u.tmp
</pre>
三、上传文件, 因为没有测试服务器,无法调试。 代码跟上传文件类似。
<pre>let uploadTask = session.uploadTask(with: request, from: data) {
(data:Data?, response:URLResponse?, error:Error?) -> Void in
//上传完毕
if error != nil{
print(error)
}else{
let str = String(data: data!, encoding: String.Encoding.utf8)
print("上传完毕:(str)") //str是包体
}
}</pre>
小结: iOS对HTTP/HTTPS交互封装个一套完整方便的API,主要涉及URLSession、URLSesionTask、URLCache及其派生类; 支持文件、上传/下载文件。