Swift - 网络 URLSession
2019-08-16 本文已影响0人
H_Zimi
问:
Swift
的网络请求怎么实现?
答:用AFNetworking
、Alamofire
这些框架啊,快速、好用、数据解析难度低。
是的,第三方网络框架真的好用,好用到我们都快忘了原生API
是怎么玩了,但万丈高楼平地起,原生API
我们还是有必要掌握的,能让我们更容易、更深入理解AFNetworking
、Alamofire
等框架,写这篇文章的目的是对Swift
的原生API
URLSession
的复习
基本网络请求:
- 先做一点准备工作,iOS从9.0版本开始添加了对应用数据传输的安全性要求(ATS),我们在这里简单的设置一下关闭ATS
- 在
info.plist
中添加一个key
:App Transport Security Settings
,类型为字典; - 再在这字典下添加另一个
key
:Allow Arbitrary Loads
,类型为Boolean
类型,值为YES
; - 网上对于解决
ATS
问题的方法有很多,这里不做赘述;
- 在
// Swift简单网络请求
let urlString = "http://rap2api.taobao.org/app/mock/228467/api/friend/list"
URLSession.shared.dataTask(with: URL(string: urlString)!) { (data, response, error) in
if error != nil {
print("\(String(describing: error))")
}
print("\(String(describing: response))")
}.resume()
//OC 网络请求
NSString *urlString = [NSString stringWithFormat:@"http://rap2api.taobao.org/app/mock/228467/api/friend/list"];
[[[NSURLSession sharedSession] dataTaskWithURL:[NSURL URLWithString:urlString] completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
if (!error) {
NSLog(@"%@",error);
}
NSLog(@"%@", response);
}] resume];
-
resume
重启连接网络请求,默认是挂起状态,而在AFNetworking
、Alamofire
中我们不需要关心这一步,它们已经帮我们做了这一步的处理; -
dataTask
创建网络会话任务-
uploadTask
文件上传任务 -
downloadTask
文件下载任务 -
streamTask
用于对通过URLSession
创建的TCP/IP
流执行读写操作
-
网络请求配置 - Configuration
一、模式
open class var `default`: URLSessionConfiguration { get }
open class var ephemeral: URLSessionConfiguration { get }
@available(iOS 8.0, *)
open class func background(withIdentifier identifier: String) -> URLSessionConfiguration
-
default
默认配置,会本地存储凭据、缓存和cookie
-
ephemeral
临时会话,不会将cookie
、缓存或凭据储存到本地只会放到内存中,当应用程序退出后数据也会消失,可以用于实现“秘密浏览”; -
background
建立后台会话,可以在应用程序挂起、退出、崩溃的情况下运行上传和下载任务,后台另起一个线程。另外,系统会根据设备的负载程度决定分配下载的资源,因此有可能会很慢甚至超时失败;-
background
需要有一个唯一标识符identifier
用于标识任务
-
后台执行任务的时候注意点
- 苹果官方文档提供的后台任务的四步操作
//Listing 1 Creating a background URL session
private lazy var urlSession: URLSession = {
let config = URLSessionConfiguration.background(withIdentifier: "MySession")
config.isDiscretionary = true
config.sessionSendsLaunchEvents = true
return URLSession(configuration: config, delegate: self, delegateQueue: nil)
}()
//Listing 2 Creating a download task from a URL session
let backgroundTask = urlSession.downloadTask(with: url)
backgroundTask.earliestBeginDate = Date().addingTimeInterval(60 * 60)
backgroundTask.countOfBytesClientExpectsToSend = 200
backgroundTask.countOfBytesClientExpectsToReceive = 500 * 1024
backgroundTask.resume()
//Listing 3 Storing the background download completion handler sent to the application delegate
var backgroundCompletionHandler: (() -> Void)?
func application(_ application: UIApplication,
handleEventsForBackgroundURLSession identifier: String,
completionHandler: @escaping () -> Void) {
backgroundCompletionHandler = completionHandler
}
//Listing 4 Executing the background URL session completion handler on the main queue
func urlSessionDidFinishEvents(forBackgroundURLSession session: URLSession) {
DispatchQueue.main.async {
guard let appDelegate = UIApplication.shared.delegate as? AppDelegate,
let backgroundCompletionHandler =
appDelegate.backgroundCompletionHandler else {
return
}
backgroundCompletionHandler()
}
}
- 给我们的信息是后台任务创建并执行还不够,还需要开启权限
- 在任务完成,系统将执行在
AppDelegate
中保存的闭包backgroundCompletionHandler
回调,更新屏幕
PS
: 官方文档查看方法:
二、请求缓存策略
public enum CachePolicy : UInt {
//默认策略
case useProtocolCachePolicy
//指定应从原始源加载URL加载的数据, 没有本地缓存数据
case reloadIgnoringLocalCacheData
//指定不仅应忽略本地缓存数据,而且应指示代理和其他中间人在协议允许的情况下忽略其缓存。未实现
case reloadIgnoringLocalAndRemoteCacheData
//与reloadIgnoringLocalCacheData相同
public static var reloadIgnoringCacheData: NSURLRequest.CachePolicy { get }
//指定应使用现有缓存数据来满足URL加载请求,不管其生命周期或到期日期。 但是,如果缓存中没有与URL加载请求相对应的现有数据,则从原始源加载URL
case returnCacheDataElseLoad
//指定应使用现有缓存数据来满足URL加载请求,不管其生命周期或到期日期。 但是,如果缓存中没有与URL加载请求相对应的现有数据,则不会尝试从源源加载URL,并且认为加载失败。 此常量指定类似于“脱机”模式的行为。
case returnCacheDataDontLoad
//指定可以使用现有缓存数据,前提是源源确认其有效性,否则从原始源加载URL。未实现。
case reloadRevalidatingCacheData
}
三、常用的属性
-
identifier
后台会话标识符 -
timeoutIntervalForRequest
请求超时,如果在设定的超时时间内没有传输数据,将导致超时,并且每当传输数据时都会重置。 -
timeoutIntervalForResource
请求超时。 如果在给定超时内无法检索到资源,则会导致超时。 -
networkServiceType
网络服务类型 -
allowsCellularAccess
是否允许蜂窝网络 -
waitsForConnectivity
任务等待网络连接是否可用,后台会话时会忽略,因为后台会话始终等待连接。
四、安全策略
-
TLSMinimumSupportedProtocol
允许最小TLS
协议版本 -
TLSMaximumSupportedProtocol
允许最大TLS
协议版本 -
urlCredentialStorage
存储身份验证凭证的库
五、HTTP
策略、代理属性
-
HTTPShouldUsePipelining
允许使用HTTP流水线 -
HTTPMaximumConnectionsPerHost
给定主机的最大并发持久连接数
六、Cookie设置
-
HTTPShouldSetCookies
是否允许请求包含一个Cookie
库中的Cookie
-
HTTPCookieAcceptPolicy
确定何时接受Cookie
策略 -
HTTPCookieStorage
保存当前会话的Cookie
的库
这里只是介绍了
Swift
的URLSession
中一些基本的属性的用途,关于URLSessionDataTask
做另一篇章来讲。
PS:鄙人小小码农一枚,限于能力,官方文档或许理解有误,文章中有错误、不足之处,欢迎各位大神指正。