Alamofire框架详细解析(二) —— 高级用法(一)

版本记录

版本号 时间
V1.0 2020.10.16 星期五

前言

关于网络请求有很多优秀的三方框架,比较常用的比如说OC的AFNetworking,这里我们就一起学习一下Swift的网络请求框架 - Alamofire。感兴趣的可以看下面几篇文章。
1. Alamofire框架详细解析(一) —— 基本概览(一)

开始

首先看下主要内容:

在本教程中,您将学习Alamofire的高级用法。 主题包括处理OAuthnetwork logging, reachability,caching等。内容来自翻译

下面看一下写作环境:

Swift 5, iOS 14, Xcode 12

接着就是正文啦。

Alamofire是最流行和广泛使用的Swift网络库之一。它建立在Apple的Foundation网络堆栈上,提供了一个优雅的API来发出网络请求。它在GitHub上拥有3万颗星,是最受好评的Swift仓库之一。

今天,您将学习并掌握它。

在本教程中,您将介绍Alamofire的高级用法。在构建GitHub客户端应用程序GitOnFire时,您将应用这些概念。在此过程中,您将学习如何:

  • 使用OAuth处理身份验证。
  • 记录网络请求和响应。
  • 重试网络请求。
  • 检查网络可达性。
  • 缓存网络响应。

这是一个有趣的事实:Alamofire以得克萨斯州官方州花之一Alamo Fire flower花的名字命名。

注意:本教程假定您熟悉Apple’s URL Loading System。如果您是Alamofire的新手,请查看Alamofire 5 For iOS: Getting Started。您还需要一个GitHub帐户。

打开入门项目。

您将使用一个名为GitOnFire的应用程序。入门项目随附Alamofire网络库和GitHub API的基本集成。

在启动文件夹中打开项目。构建并运行。

您会在GitHub中看到最受欢迎的Swift存储库列表。 注意Alamofire在该列表上的位置。

指定搜索文本以获取热门存储库的相关列表。 点击存储库的名称以查看其最近的提交。

您将更新网络调用以使用自定义URLSessionConfiguration。 您还将集成OAuth登录并从您的帐户中获取存储库。

这些是您首先要处理的文件:

  • GitAPIManager.swift:使用AlamofireGitHub获取存储库并提交。
  • RepositoriesViewController.swift:显示存储库列表。
  • LoginViewController.swift:处理GitHub OAuth登录。

在本教程中,您将使用GitHub API处理OAuth身份验证,获取仓库和提交。 有关所有GitHub API及其用途的详尽列表,请参阅此API documentation


Custom Session and URLSessionConfiguration

打开GitAPIManager.swift并找到searchRepositories(query:completion :)。 搜索GitHub存储库的实现已经到位。 您将利用这个机会来了解方法的内部原理。

func searchRepositories(
  query: String,
  completion: @escaping ([Repository]) -> Void
) {
  // 1
  let url = "https://api.github.com/search/repositories"
  // 2
  var queryParameters: [String: Any] = [
    "sort": "stars",
    "order": "desc",
    "page": 1
  ]
  queryParameters["q"] = query
  // 3
  AF.request(url, parameters: queryParameters).responseDecodable(
    of: Repositories.self) { response in
      guard let items = response.value else {
        return completion([])
      }
      completion(items.items)
    }
}

以下是分步细分:

  • 1) 您定义URL以搜索存储库。
  • 2) 然后,创建查询参数数组以获取存储库。 您可以根据存储库的星数指定降序排序。
  • 3) 接下来,您使用Alamofire的默认会话AF发出网络请求。 该方法对收到的响应进行解码,并在完成块中以自定义模型Repository的数组形式返回。

每个请求的默认超时为60秒。 要将超时间隔更新为30秒,可以将requestModifier指定为请求的一部分,如下所示:

AF.request(url, parameters: queryParameters) { urlRequest in
  urlRequest.timeoutInterval = 30
}

您可以在构造请求时将requestModifier指定为尾随block。 但是,如果所有会话请求都需要超时间隔,该怎么办? 您使用自定义URLSessionConfiguration

1. Customizing Session

Alamofire提供了Session,在责任方面类似于URLSession。 它有助于创建和管理不同的请求,并为所有请求提供通用功能,例如侦听,响应缓存等。 在本教程的后面,您将了解有关不同类型请求的更多信息。

您可以通过将URLSessionConfiguration与所需的配置一起使用来自定义会话行为。 将以下代码添加到GitAPIManager:

//1
let sessionManager: Session = {
  //2
  let configuration = URLSessionConfiguration.af.default
  //3
  configuration.timeoutIntervalForRequest = 30
  //4
  return Session(configuration: configuration)
}()

在这里:

  • 1) 将sessionManager定义为自定义Session
  • 2) 然后从URLSessionConfiguration.af.default创建URLSessionConfiguration的实例。 默认的Alamofire URLSessionConfiguration添加了Accept-Encoding,Accept-LanguageUser-Agent标头。
  • 3) 将配置上的timeoutIntervalForRequest设置为30秒,这适用于该会话中的所有请求。
  • 4) 定义配置后,您可以通过传入自定义配置来创建Session的实例。

现在,您将使用此sessionManager处理所有网络请求。 在searchRepositories(query:completion :)中,将下面:

AF.request(url, parameters: queryParameters)

替换为

sessionManager.request(url, parameters: queryParameters)

在这里,您使用sessionManager而不是AF发出请求。

接下来在fetchCommits(for:completion :)中,替换:

AF.request(url)

sessionManager.request(url)

现在,两个网络请求的请求超时间隔都设置为30秒。

接下来,您将看到使用自定义会话的优势。

sessionManager中,在configuration.timeoutIntervalForRequest = 30下面添加以下代码:

configuration.waitsForConnectivity = true

在这里,将waitsForConnectivity设置为true,这使会话在发出请求之前先等待网络连接。 您必须为URLSessionConfiguration设置所有自定义配置,然后才能将它们添加到Session中。 如果您在将配置属性添加到会话Session后对其进行了更改,则它们不会产生任何影响。

查看实际的代码。 关闭WiFi。 构建并运行。

该应用会加载,并显示一个带有加载指示器的空白列表。 然后,打开WiFi。 在几秒钟内,存储库将加载。 感觉像魔术吗?

在本教程的稍后部分,您将学习如何监视和处理网络可达性。

接下来,您将学习如何使用事件监视器Event Monitor记录网络请求和响应。


Logging Network Requests and Responses Using Event Monitor

到目前为止,您已经发出网络请求以获取存储库和提交,并将结果显示在table view中。 很好,但是您可以走得更远,看到原始的网络请求和响应。

Alamofire提供了一种通过EventMonitor协议深入了解所有内部事件的强大方法。 EventMonitor包含多个Alamofire事件,例如URLSessionDelegate请求事件。 这使EventMonitor非常适合记录事件。

Networking组中创建一个名为GitNetworkLogger.swift的新Swift文件。 添加以下代码:

import Alamofire

class GitNetworkLogger: EventMonitor {
  //1
  let queue = DispatchQueue(label: "com.raywenderlich.gitonfire.networklogger")
  //2
  func requestDidFinish(_ request: Request) {
    print(request.description)
  }
  //3
  func request<Value>(
    _ request: DataRequest,
    didParseResponse response: DataResponse<Value, AFError>
  ) {
    guard let data = response.data else {
      return
    }
    if let json = try? JSONSerialization
      .jsonObject(with: data, options: .mutableContainers) {
        print(json)
    }
  }
}

以下是代码细分:

  • 1) EventMonitor需要一个DispatchQueue来调度所有事件。 默认情况下,EventMonitor使用主队列。

您可以使用自定义DispatchQueue初始化queue以保持性能。 这是一个串行队列,可以处理会话中的所有事件。

  • 2) 请求完成时将调用requestDidFinish(_ :)。 然后,您打印请求的description以在控制台中查看HTTP方法和请求URL
  • 3) 收到响应时将调用request(_:didParseResponse :)。 使用JSONSerialization,您可以将响应呈现为JSON,然后将其打印到控制台。

打开GitAPIManager.swift。 在sessionManager中的configuration.waitsForConnectivity = true下面添加以下代码:

let networkLogger = GitNetworkLogger()

在这里,您将networkLogger定义为GitNetworkLogger的实例。

现在,将return Session(configuration:configuration)替换为以下内容:

return Session(configuration: configuration, eventMonitors: [networkLogger])

Session初始化期间,将networkLogger以数组形式传递给eventMonitors。 构建并运行。

现在,您将在控制台中看到所有网络请求和响应记录。很好!

到目前为止,您已经获取了公共存储库。现在是时候从您自己的GitHub帐户中获取存储库了。准备一些授权乐趣。


GitHub Authorization

要获取您的私有存储库,您需要通过您的应用程序登录GitHub。应用可以通过两种方式获得访问GitHub API的授权:

  • Basic Authentication - 基本身份验证:这涉及作为请求的一部分传递的用户名和密码。
  • OAuth 2.0 tokenOAuth 2.0是一个授权框架,可为应用程序提供对HTTP服务用户帐户的访问权限。

在本教程中,您将学习如何使用OAuth 2.0token

1. OAuth Overview

授权应用程序通过OAuth 2.0访问用户存储库的步骤如下:

  • 1) 该应用发出网络授权请求。
  • 2) 然后,用户登录GitHub以获得成功的授权。
  • 3) 接下来,GitHub使用临时码重定向回应用程序。
  • 4) 该应用使用该临时码请求token
  • 5) 收到访问token后,该应用会发出API请求以获取用户的私有存储库。请求的授权标头将包含访问token

接下来,您将创建一个GitHub OAuth应用。

2. Creating GitHub OAuth App

登录GitHub并按照以下步骤steps创建具有以下设置的OAuth应用:

  • 1) 输入GitOnFire作为应用程序名称。
  • 2) 输入https://www.raywenderlich.com/作为主页URL。
  • 3) 跳过应用程序描述。
  • 4) 输入gitonfire://作为授权回调URL。

3. Logging Into GitHub

注册应用程序后,复制Client IDClient Secret值。 然后在您的Xcode项目中,打开GitHubConstants.swift并使用相应的值更新clientIDclientSecret

接下来,打开GitAPIManager.swift并在右括号之前添加以下方法:

func fetchAccessToken(
  accessCode: String,
  completion: @escaping (Bool) -> Void
) {
  // 1
  let headers: HTTPHeaders = [
    "Accept": "application/json"
  ]
  // 2
  let parameters = [
    "client_id": GitHubConstants.clientID,
    "client_secret": GitHubConstants.clientSecret,
    "code": accessCode
  ]
  // 3
  sessionManager.request(
    "https://github.com/login/oauth/access_token",
    method: .post,
    parameters: parameters,
    headers: headers)
    .responseDecodable(of: GitHubAccessToken.self) { response in
      guard let cred = response.value else {
        return completion(false)
      }
      TokenManager.shared.saveAccessToken(gitToken: cred)
      completion(true)
    }
}

以下是分步细分:

  • 1) 您定义请求的标头。带有application / jsonAccept告诉服务器该应用想要JSON格式的响应。
  • 2) 然后定义查询参数client_id,client_secretcode。 这些参数作为请求的一部分发送。
  • 3) 您发出网络请求以获取访问token。 响应被解码为GitHubAccessTokenTokenManager实用程序类有助于将token存储在钥匙串中。

要了解有关使用钥匙串和存储安全信息的更多信息,请阅读KeyChain Services API Tutorial for Passwords in Swift

打开LoginViewController.swift。 在getGitHubIdentity()中,替换//TODO: Call to fetch access token will be added here为以下内容:

GitAPIManager.shared.fetchAccessToken(accessCode: value) { [self] isSuccess in
  if !isSuccess {
    print("Error fetching access token")
  }
  navigationController?.popViewController(animated: true)
}

在这里,您使用临时码进行调用以获取访问token。 响应成功后,控制器将显示存储库列表。

现在打开RepositoriesViewController.swift。 在viewDidLoad()中,删除以下行:

loginButton.isHidden = true

这将显示登录按钮。 构建并运行。

点击Login以登录。浏览器随后将您重定向到该应用程序,并且“登录”按钮将变为注销。 您会在控制台中看到访问access tokenscope

很好! 现在是时候获取您的存储库了。


Fetching User Repositories

打开GitAPIManager.swift。 在GitAPIManager中,添加以下方法:

func fetchUserRepositories(completion: @escaping ([Repository]) -> Void) {
  //1
  let url = "https://api.github.com/user/repos"
  //2
  let parameters = ["per_page": 100]
  //3
  sessionManager.request(url, parameters: parameters)
    .responseDecodable(of: [Repository].self) { response in
      guard let items = response.value else {
        return completion([])
      }
      completion(items)
    }
}

这是您添加的内容:

  • 1) 您定义URL来获取您的存储库。
  • 2) per_page查询参数确定每个响应返回的最大存储库数。 每页最多可获取100个结果。
  • 3) 接下来,您请求获取存储库。 然后,您将响应解码为Repository数组,并将其传递到完成块中。

接下来,打开RepositoriesViewController.swift并找到fetchAndDisplayUserRepositories()。 替换//TODO: Add more here..为:

//1
loadingIndicator.startAnimating()
//2
GitAPIManager.shared.fetchUserRepositories { [self] repositories in
  //3
  self.repositories = repositories
  loadingIndicator.stopAnimating()
  tableView.reloadData()
}

以下是代码细分:

  • 1) 在发出网络请求之前,您会显示一个加载指示器。
  • 2) 然后,您发出网络请求以获取您的存储库。
  • 3) 提取存储库后,即可使用响应设置repositories并关闭加载指示符。 然后,您重新加载table view以显示存储库。

默认情况下,Alamofire在主队列上调用响应处理程序。 因此,您无需添加代码即可切换到主线程来更新UI。

构建并运行。

清单是空的! 检查Xcode控制台,您会看到401未经授权的请求。

您必须在标头中传递access token以进行授权。 您可以在GitAPIManager的fetchUserRepositories(completion :)内添加一个Authentication标头。 但是,为每个请求单独添加标头的过程可能会重复。

为了避免这种情况,Alamofire提供了RequestInterceptor,该协议可启用强大的按会话和按请求的功能。


Request Overview

在深入研究RequestInterceptor之前,您应该了解Request的不同类型。

AlamofireRequest是所有请求的超类。 有几种类型:

  • DataRequest:通过将服务器响应下载到存储在内存中的数据中来封装URLSessionDataTask
  • DataStreamRequest:封装URLSessionDataTask并随时间流化来自HTTP连接的数据。
  • UploadRequest:封装URLSessionUploadTask并将数据上传到远程服务器。
  • DownloadRequest:通过将响应数据下载到磁盘来封装URLSessionDownloadTask

每个请求均以initialized状态开始。 它可以在其生命周期中被suspended, resumedcanceled。 请求以finished状态结束。

目前,您正在使用DataRequest来获取存储库。 现在,您将使用RequestInterceptor拦截请求。


RequestInterceptor Overview

AlamofireRequestInterceptor包含两个协议:RequestAdapterRequestRetrier

通过RequestAdapter,您可以在发送每个请求之前对其进行检查和更改。 当每个请求都包含一个Authorization header时,这是理想的选择。

RequestRetrier重试遇到错误的请求。

1. Integrating RequestInterceptor

Networking中,创建一个名为GitRequestInterceptor.swift的新Swift文件。 打开文件并添加:

import Alamofire

class GitRequestInterceptor: RequestInterceptor {
  //1
  let retryLimit = 5
  let retryDelay: TimeInterval = 10
  //2
  func adapt(
    _ urlRequest: URLRequest,
    for session: Session,
    completion: @escaping (Result<URLRequest, Error>) -> Void
  ) {
    var urlRequest = urlRequest
    if let token = TokenManager.shared.fetchAccessToken() {
      urlRequest.setValue("token \(token)", forHTTPHeaderField: "Authorization")
    }
    completion(.success(urlRequest))
  }
  //3
  func retry(
    _ request: Request,
    for session: Session,
    dueTo error: Error,
    completion: @escaping (RetryResult) -> Void
  ) {
    let response = request.task?.response as? HTTPURLResponse
    //Retry for 5xx status codes
    if 
      let statusCode = response?.statusCode,
      (500...599).contains(statusCode),
      request.retryCount < retryLimit {
        completion(.retryWithDelay(retryDelay))
    } else {
      return completion(.doNotRetry)
    }
  }
}

以下是分步细分:

  • 1) 您声明两个常量,retryLimitretryDelay。它们有助于对重试请求的尝试次数和重试尝试之间的持续时间施加限制。
  • 2) RequestAdapterRequestInterceptor的一部分。它只有一个条件,即adapt(_:for:completion :)

该方法检查并调整请求。由于completion handler是异步的,因此此方法可以在发出请求之前从网络或磁盘中获取token

在这里,您从钥匙串中获取token并将其添加到Authorization header中。 GitHub OAuth Appsaccess token没有到期时间。但是,授权该应用程序的用户可以通过GitHub设置将其撤消。

  • 3) RequestRetrier有一个要求,即retry(_:for:dueTo:completion :)。当请求遇到错误时,将调用该方法。您必须使用RetryResult调用completion block,以指示是否应重试该请求。

在这里,您检查响应码是否包含5xx错误码。当服务器无法满足有效请求时,它将返回5xx码。例如,当服务关闭以进行维护时,您可能会获得503错误代码。

如果错误包含5xx错误代码,则请求将以retryDelay中指定的延迟重试,前提是计数在retryLimit之内。

打开GitAPIManager.swift。在sessionManagerlet networkLogger = GitNetworkLogger()下面添加以下代码:

let interceptor = GitRequestInterceptor()

在这里,您将interceptor定义为GitRequestInterceptor的实例。 在sessionManager中将Session初始化替换为以下内容:

return Session(
  configuration: configuration,
  interceptor: interceptor,
  eventMonitors: [networkLogger])

使用此代码,您可以在Session的构造函数中传递新创建的interceptor。 现在,通过GitRequestInterceptor实例拦截了属于sessionManager的所有请求。 构建并运行。

现在,您将看到从GitHub帐户获取的存储库。


Routing Requests and URLRequestConvertible

到目前为止,在发出网络请求时,您已经为每个请求提供了URL路径,HTTP方法和查询参数。 随着应用程序大小的增长,必须使用一些常见的模式来构建网络堆栈。 Router设计模式通过定义每个请求的路由和组件来提供帮助。

Networking中,打开GitRouter.swift。 您将看到到目前为止所提出的所有请求,并在枚举中捕获为不同的情况。 使用此GitRouter构造您的请求。

将以下扩展名添加到GitRouter.swift的末尾:

//1
extension GitRouter: URLRequestConvertible {
  func asURLRequest() throws -> URLRequest {
    //2
    let url = try baseURL.asURL().appendingPathComponent(path)
    var request = URLRequest(url: url)
    request.method = method
    //3
    if method == .get {
      request = try URLEncodedFormParameterEncoder()
        .encode(parameters, into: request)
    } else if method == .post {
      request = try JSONParameterEncoder().encode(parameters, into: request)
      request.setValue("application/json", forHTTPHeaderField: "Accept")
    }
    return request
  }
}

这是一个细分:

  • 1) 您将扩展名添加到GitRouter以符合URLRequestConvertible。 该协议有一个要求asURLRequest(),它有助于构造URLRequest。 符合URLRequestConvertible有助于抽象并确保请求的端点的一致性。
  • 2) 在这里,您可以使用GitRouter中的属性构造请求。
  • 3) 基于HTTP方法,您可以使用URLEncodedFormParameterEncoderJSONParameterEncoder对参数进行编码。 Accept HTTP标头设置为POST请求。 您在构造请求后返回该请求。

现在,打开GitAPIManager.swift。 您将更新所有请求方法以使用GitRouter

fetchCommits(for:completion :)中,删除以let url =开头的行。 现在,将以下内容替换为sessionManager.request(url)以使用新路由器:

sessionManager.request(GitRouter.fetchCommits(repository))

searchRepositories(query:completion :)中,删除sessionManager.request ...之前的所有内容。现在,将sessionManager.request(url,parameters:queryParameters)替换为:

sessionManager.request(GitRouter.searchRepositories(query))

同样,在fetchAccessToken(accessCode:completion :)中,删除sessionManager.request ...之前的所有内容。现在,将sessionManager.request(...)调用替换为:

sessionManager.request(GitRouter.fetchAccessToken(accessCode))

最后,在fetchUserRepositories(completion :)中,删除sessionManager.request(url,parameters:parameters)之前的所有内容,并将该行替换为:

sessionManager.request(GitRouter.fetchUserRepositories)

您将删除在每个方法中本地声明的URL,查询参数和标头,因为您不再需要它们。GitRouter为每个请求构造URLRequests。 构建并运行。

除了基础请求使用GitRouter之外,您将看到存储库像以前一样加载。

到目前为止,该应用程序运行良好。 有了良好的网络,结果几乎是即时的。 但是网络是不可预测的野兽。

了解何时无法访问网络并在您的应用中通知用户,这一点很重要。 AlamofireNetworkReachabilityManager随时为您服务!


Network Reachability

Networking组中,打开GitNetworkReachability.swift。 在右括号之前添加以下内容:

// 1
let reachabilityManager = NetworkReachabilityManager(host: "www.google.com")
// 2
func startNetworkMonitoring() {
  reachabilityManager?.startListening { status in
    switch status {
    case .notReachable:
      self.showOfflineAlert()
    case .reachable(.cellular):
      self.dismissOfflineAlert()
    case .reachable(.ethernetOrWiFi):
      self.dismissOfflineAlert()
    case .unknown:
      print("Unknown network state")
    }
  }
}

GitNetworkReachability提供了一个单例。 它包括显示和关闭弹窗的功能。 这是您添加的内容:

  • 1) AlamofireNetworkReachabilityManager侦听host和地址的可达性。 它适用于蜂窝和WiFi网络接口。 在这里,您将创建一个属性reachabilityManager,作为NetworkReachabilityManager的实例。 这会使用www.google.com作为host检查可访问性。
  • 2) startNetworkMonitoring()侦听网络可达性状态的更改。 如果无法访问网络,则显示弹窗。 一旦存在通过任何网络接口可访问的网络,弹窗将被消除。

现在,打开AppDelegate.swift。 在return true之前,在application(_:didFinishLaunchingWithOptions :)中添加以下内容:

GitNetworkReachability.shared.startNetworkMonitoring()

在这里,您可以在GitNetworkReachability上调用startNetworkMonitoring()以在应用启动时开始侦听网络可达性状态。

构建并运行。 应用启动后,请关闭网络。

该应用程序会在无法访问网络时向用户显示弹窗,并在可访问网络时将其关闭。 以用户为中心,真是太好了!

注意:您应该在真实设备上测试与网络可达性相关的功能,因为可达性可能无法在模拟器上正常工作。 您可以通过阅读此GitHub post来了解有关此问题的更多信息。

有时显示弹窗不是理想的体验。 相反,您可能希望在没有网络时显示以前获取的应用程序数据。 AlamofireResponseCacher在这里为您提供帮助。


Caching Using ResponseCacher

打开GitAPIManager.swift。 在sessionManager中删除以下配置选项:

configuration.timeoutIntervalForRequest = 30
configuration.waitsForConnectivity = true

在这里,您删除了timeoutIntervalForRequestwaitsForConnectivity配置选项,因此该应用程序无需等待网络连接。

sessionManagerlet configuration = URLSessionConfiguration.af.default下面添加:

//1
configuration.requestCachePolicy = .returnCacheDataElseLoad
//2
let responseCacher = ResponseCacher(behavior: .modify { _, response in
  let userInfo = ["date": Date()]
  return CachedURLResponse(
    response: response.response,
    data: response.data,
    userInfo: userInfo,
    storagePolicy: .allowed)
})

这是一个细分:

  • 1) 要缓存会话请求,请在URLSessionConfiguration上将requestCachePolicy设置为returnCacheDataElseLoad。 设置后,缓存将返回响应。 如果缓存没有响应,则会发出网络请求。
  • 2) AlamofireResponseCacher使您可以轻松地指定在存储到缓存之前是否需要缓存,不缓存或修改请求。 在这里,您可以通过指定.modify来修改响应,然后再将其保存到缓存中。 通过将响应的日期传递到userInfo字典中,可以保存响应的日期以及响应的日期。

现在,如下所示在sessionManager中更新Session初始化:

return Session(
  configuration: configuration,
  interceptor: interceptor,
  cachedResponseHandler: responseCacher,
  eventMonitors: [networkLogger])

在这里,您将Session的构造函数中的responseCacher作为cachedResponseHandler传递。 这使responseCacher处理Session中所有请求的缓存行为。

打开AppDelegate.swift。 注释掉以下可启动网络监视的代码行:

GitNetworkReachability.shared.startNetworkMonitoring()

这样可以防止应用程序离线时显示No Network弹窗。 关闭模拟器或设备上的网络访问。 构建并运行。

您将看到从缓存加载的存储库。

恭喜你! 您现在是Alamofire专业人士。

在此Alamofire教程中,您学习了如何:

  • 创建一个自定义SessionURLSessionConfiguration
  • 使用EventMonitor记录网络请求和响应。
  • 使用RequestInterceptor处理身份验证和重试。
  • 使用URLRequestConvertible配置路由器。
  • 使用NetworkReachabilityManager检查网络可达性。
  • 使用ResponseCacher在进行缓存之前修改响应。

要了解更多信息,请查看Alamofire高级用法文档Alamofire advanced usage documentation

后记

本篇主要讲述了Alamofire框架的基本概览,感兴趣的给个赞或者关注~~~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,377评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,390评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,967评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,344评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,441评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,492评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,497评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,274评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,732评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,008评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,184评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,837评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,520评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,156评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,407评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,056评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,074评论 2 352