Swift - 网络URLSession

问:Swift的网络请求怎么实现? 答:用AFNetworkingAlamofire这些框架啊,快速、好用、数据解析难度低。 是的,第三方网络框架真的好用,好用到咱们都快忘了原生API是怎么玩了,但万丈高楼平地起,原生API咱们仍是有必要掌握的,能让咱们更容易、更深刻理解AFNetworkingAlamofire等框架,写这篇文章的目的是对Swift的原生API URLSession的复习api

基本网络请求:

  • 先作一点准备工做,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重启链接网络请求,默认是挂起状态,而在AFNetworkingAlamofire中咱们不须要关心这一步,它们已经帮咱们作了这一步的处理;
  • dataTask建立网络会话任务
    • uploadTask文件上传任务
    • downloadTask文件下载任务
    • streamTask用于对经过URLSession建立的TCP/IP流执行读写操做

网络请求配置 - Configuration

1、模式
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: 官方文档查看方法: 安全

2、请求缓存策略
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
}
复制代码
3、经常使用的属性
  • identifier后台会话标识符
  • timeoutIntervalForRequest请求超时,若是在设定的超时时间内没有传输数据,将致使超时,而且每当传输数据时都会重置。
  • timeoutIntervalForResource请求超时。 若是在给定超时内没法检索到资源,则会致使超时。
  • networkServiceType网络服务类型
  • allowsCellularAccess是否容许蜂窝网络
  • waitsForConnectivity任务等待网络链接是否可用,后台会话时会忽略,由于后台会话始终等待链接。
4、安全策略
  • TLSMinimumSupportedProtocol容许最小TLS协议版本
  • TLSMaximumSupportedProtocol容许最大TLS协议版本
  • urlCredentialStorage存储身份验证凭证的库
5、HTTP策略、代理属性
  • HTTPShouldUsePipelining容许使用HTTP流水线
  • HTTPMaximumConnectionsPerHost给定主机的最大并发持久链接数
6、Cookie设置
  • HTTPShouldSetCookies是否容许请求包含一个Cookie库中的Cookie
  • HTTPCookieAcceptPolicy肯定什么时候接受Cookie策略
  • HTTPCookieStorage 保存当前会话的 Cookie的库

这里只是介绍了SwiftURLSession中一些基本的属性的用途,关于URLSessionDataTask作另外一篇章来说。bash

PS:鄙人小小码农一枚,限于能力,官方文档或许理解有误,文章中有错误、不足之处,欢迎各位大神指正。cookie

相关文章
相关标签/搜索