Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >iOS开发之网络代码进化史

iOS开发之网络代码进化史

作者头像
YungFan
发布于 2025-03-11 01:49:39
发布于 2025-03-11 01:49:39
6300
代码可运行
举报
文章被收录于专栏:学海无涯学海无涯
运行总次数:0
代码可运行

介绍

目前,iOS 原生网络请求数据共有 4 种方式,分别是传统 Completion 回调Completion 回调 + ResultCombine 框架Swift Concurrency (async/await)。本文以下载图片为例,详细讲解 4 种网络请求的差异。

传统Completion回调

一种最基础的异步处理方式,使用闭包作为回调函数。

代码

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
/// 错误枚举
enum FetchError: Error {
    case badURL
    case badRequest
    case badImage
}

/// 1. 传统completion回调
func fetchImage(from url: String, completion: @escaping (UIImage?, Error?) -> Void) {
    guard let url = URL(string: url) else {
        completion(nil, FetchError.badURL)
        return
    }
    let request = URLRequest(url: url)
    let task = URLSession.shared.dataTask(with: request) { data, response, error in
        if let error = error {
            completion(nil, error)
        } else if (response as? HTTPURLResponse)?.statusCode != 200 {
            completion(nil, FetchError.badRequest)
        } else {
            guard let data = data,
                  let image = UIImage(data: data) else {
                completion(nil, FetchError.badImage)
                return
            }
            completion(image, nil)
        }
    }
    task.resume()
}

fetchImage(from: "https://search-operate.cdn.bcebos.com/7e85570b817e17e8f3ae93134cc78451.gif") { image, error in
    if let image {
        print(image)
    } else if let error {
        print(error)
    }
}

优点

  • 简单易懂,学习曲线低。
  • 兼容性好,适用于所有 iOS 版本。

缺点

  • 可能导致"回调地狱"。
  • 错误处理比较分散,每个回调都需要单独处理错误。
  • 取消操作比较麻烦。

Completion回调+Result

对传统 Completion 回调的改进,使用 Swift 的 Result 类型统一处理成功和失败情况。

代码

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
enum FetchError: Error {
    case badURL
    case badRequest
    case badImage
}

/// 2. completion回调+Result
func fetchImage(from url: String, completion: @escaping (Result<UIImage, Error>) -> Void) {
    guard let url = URL(string: url) else {
        completion(.failure(FetchError.badURL))
        return
    }
    let request = URLRequest(url: url)
    let task = URLSession.shared.dataTask(with: request) { data, response, error in
        if let error = error {
            completion(.failure(error))
        } else if (response as? HTTPURLResponse)?.statusCode != 200 {
            completion(.failure(FetchError.badRequest))
        } else {
            guard let data = data,
                  let image = UIImage(data: data) else {
                completion(.failure(FetchError.badImage))
                return
            }
            completion(.success(image))
        }
    }
    task.resume()
}

fetchImage(from: "https://search-operate.cdn.bcebos.com/7e85570b817e17e8f3ae93134cc78451.gif") { result in
    switch result {
    case let .success(image):
        print(image)
    case let .failure(error):
        print(error)
    }
}

优点

  • 相比传统回调,代码更加清晰和统一。
  • 能够明确区分成功和失败状态。
  • 兼容所有 iOS 版本。

缺点

  • 仍然会导致回调嵌套问题。
  • 取消机制同样需要额外处理。

Combine框架

Combine 是 Apple 在 iOS 13 引入的响应式编程框架,专为处理异步事件流设计。

代码

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
enum FetchError: Error {
    case badURL
    case badRequest
    case badImage
}

/// 3. Combine
func fetchImage(from url: String) -> AnyPublisher<UIImage, Error> {
    guard let url = URL(string: url) else { return Fail<UIImage, Error>(error: FetchError.badURL).eraseToAnyPublisher() }
    let request = URLRequest(url: url)
    let session = URLSession.shared
    let dataPublisher = session.dataTaskPublisher(for: request)
        .retry(3)
        .timeout(5, scheduler: DispatchQueue.global())
        .tryMap { data, response -> Data in
            guard (response as? HTTPURLResponse)?.statusCode == 200 else { throw FetchError.badRequest }
            return data
        }
        .tryMap { data -> UIImage in
            guard let image = UIImage(data: data) else {throw FetchError.badImage}
            return image
        }
        .receive(on: DispatchQueue.main)
        .subscribe(on: DispatchQueue.global())
        .eraseToAnyPublisher()
    return dataPublisher
}

var cancellables = Set<AnyCancellable>()
fetchImage(from: "https://search-operate.cdn.bcebos.com/7e85570b817e17e8f3ae93134cc78451.gif")
    .sink(
        receiveCompletion: { completion in
            switch completion {
            case .finished:
                print("Image download completed successfully")
            case let .failure(error):
                print(error)
            }
        }, receiveValue: { image in
            print(image)
        }
    )
    .store(in: &cancellables)

优点

  • 声明式编程风格,代码更简洁。
  • 强大的操作符支持。
  • 优雅的错误处理和转换机制。
  • 提供标准的取消机制。

缺点

  • 学习曲线较陡峭。
  • 仅支持 iOS 13 及以上版本。
  • 长链式调用可能影响代码可读性。

4. Swift Concurrency (async/await)

Swift 5.5 引入的现代异步编程模型,使用结构化并发的概念。

代码

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
enum FetchError: Error {
    case badURL
    case badRequest
    case badImage
}

/// 4. Concurrency
func fetchImage(from url: String) async throws -> UIImage {
    guard let url = URL(string: url) else { throw FetchError.badURL }
    let request = URLRequest(url: url)
    let (data, response) = try await URLSession.shared.data(for: request)
    guard (response as? HTTPURLResponse)?.statusCode == 200 else { throw FetchError.badRequest }
    let image = UIImage(data: data)
    guard let image = image else { throw FetchError.badImage }
    return image
}

Task {
    do {
        let image = try await fetchImage(from: "https://search-operate.cdn.bcebos.com/7e85570b817e17e8f3ae93134cc78451.gif")
        print(image)
    } catch {
        print(error)
    }
}

优点

  • 代码简洁直观,接近同步代码的阅读体验。
  • 易于理解的错误处理。
  • 内置取消支持和结构化并发。
  • 易于调试和维护。

缺点

  • 一定的学习曲线。
  • 需要 iOS 15 及以上版本。
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2025-03-10,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
Swift3.0 Alamofire初级应用
序:做过iOS开发的人肯定都知道AFN,现在Swift逐渐流行AFN团队又用Swift写了Alamofire。从头开始学习一下,看看新的Alamofire有什么性能的优化和不同。
星宇大前端
2022/05/06
2K0
AlamofireImage 源码阅读
在AlamofireImage中一共就只有5个类加一些扩展 // 错误处理类,继承自Error,主要有requestCancelled(请求取消)、imageSerializationFailed(请求失败)两种错误 AFIError // 定义图片对象,主要用来适配mac(NSImage)和ios(UIImage)平台 Image // 图片内存缓存对象 ImageCache // 图片下载对象(下载基于Alamofire) ImageDownloader // 图片滤镜对象(CoreGraphics切圆
用户2215591
2018/06/13
1.5K0
iOS14开发-网络
上传数据需要服务端配合,不同的服务端代码可能会不一样,下面的上传代码适用于本人所写的服务端代码。
YungFan
2021/06/24
2.5K0
iOS 面试策略之系统框架-网络、推送与数据处理
如果说移动时代的前身是什么,我想一个可能的答案就是网络时代。网络的兴起,让所有设备相连成为了可能,也催生了电商、社交、搜索等多个领域的商业巨头。而移动时代,则是网络时代的必然延伸,它代表着更便捷、更广阔、更深入的连接。
会写bug的程序员
2021/05/15
1.9K0
iOS 面试策略之系统框架-网络、推送与数据处理
优雅的处理网络数据,你真的会吗?不如看看这篇.
相信大家平时在用 App 的时候, 往往有过这样的体验,那就是加载网络数据等待的时间过于漫长,滚动浏览时伴随着卡顿,甚至在没有网络的情况下,整个应用处于不可用状态。那么我们该怎么去提高用户体验,保证用
HelloWorld杰少
2022/08/04
1.4K0
优雅的处理网络数据,你真的会吗?不如看看这篇.
从AFNetworking源码分析到应用全解 原
    AFNetworking是iOS/OS开发中常用的一个第三方网络库,可以说它是目前最流行的网络库,但其代码结构其实并不复杂,也可以说非常简洁优美。在AFNetworking中,大量使用的线程安全的开发技巧,读此源码也是一次很好的多线程学习机会。本篇博客从主要结构和网络请求的主流程进行分享,解析了AFNetworking的设计思路与工作原理,后面还有其中提供的UI扩展包的接口应用总结。
珲少
2018/08/15
3.2K0
从AFNetworking源码分析到应用全解
                                                                            原
使用Swift模拟用户登录当网获取数据并保存到MySQL中
当当网作为中国最大的综合性网上商城之一,通过爬取当当网数据,我们可以获取商品信息、用户评价、销售数据等宝贵的信息资源。这些数据可以帮助企业了解市场趋势、分析竞争对手、优化产品定价等,从而做出更明智的决策。
小白学大数据
2023/10/30
2430
iOS开发·网络请求方法总结复习(NSURLConnection,NSURLSession,AFNetworking)
iOS的开发中的网络下载方式包括NSData(最原始,实际开发基本不会用),NSURLConnection(古老又过气的苹果原生网络框架),NSURLSession(现在流行的苹果网络框架),AFNetworking,SDWebImage以及基于AFNetworking的二次封装框架例如XMNetworking,HYBNetworking等等。
陈满iOS
2018/09/10
3.5K0
iOS开发·网络请求方法总结复习(NSURLConnection,NSURLSession,AFNetworking)
在iPhone上构建自定义数据采集完整指南
在iPhone上构建自定义数据采集工具可以帮助我们更好地满足特定需求,提高数据采集的灵活性和准确性。本文将为您提供一份完整的指南和示例代码,教您如何在iPhone上构建自定义数据采集工具。
华科云商小徐
2023/09/06
2190
[iOS源码笔记]·第三方网络下载处理框架:AFNetworking网络下载处理(调用栈与缓存策略篇)
看到这里,或许可以看manager对象的AFHTTPSessionManager类型更深层定义。
陈满iOS
2018/09/10
1.4K0
[iOS源码笔记]·第三方网络下载处理框架:AFNetworking网络下载处理(调用栈与缓存策略篇)
了解 Swift 的 Result 类型
通常希望函数成功返回一些数据,或者如果失败则返回错误。我们通常使用throwing函数对此建模,因为如果函数调用成功,我们将获得数据,但是如果抛出错误,则将运行catch代码块,因此我们可以独立处理这两个函数。但是,如果函数调用没有立即返回怎么办?
韦弦zhy
2020/09/14
2.7K0
了解 Swift 的 Result 类型
译文: iOS Unit Testing and UI Testing Tutorial
原文: iOS Unit Testing and UI Testing Tutorial,作者:Audrey Tam。更新于2017年3月13日。以下为正文:
iOS Development
2019/02/14
1.4K0
Combine-Subscriber
Publisher 根据 Subscriber 的请求提供数据。如果没有任何订阅请求,Publisher 不会提供任何数据。所以可以这样说,Subscriber负责向 Publisher 请求数据并接收数据(或失败)。
YungFan
2020/04/16
9970
Combine-Subscriber
面向协议编程与 Cocoa 的邂逅 (下)
本文笔者在 MDCC 16 (移动开发者大会) 上 iOS 专场中的主题演讲的文字整理。发布于 2016年11月29日 最后更新于 2020年10月22日
Swift社区
2021/11/26
7500
Swift中的网络代理设置与数据传输
网络代理(Proxy)是一种位于客户端和目标服务器之间的中间服务器,它能够转发客户端的请求到目标服务器,并将服务器的响应返回给客户端。通过使用网络代理,客户端可以间接地访问目标服务器,从而实现数据传输。网络代理的主要作用包括:
小白学大数据
2025/01/08
900
架构之路 (六) —— VIPER架构模式(二)
源码 1. Swift 首先看下工程组织结构 下面就是源码了 1. SceneDelegate.swift import SwiftUI class SceneDelegate: UIRespond
conanma
2021/09/04
1.2K0
Swift 中的 Task
Swift 中的 Task 是 WWDC 2021 引入的并发框架的一部分。任务允许我们从非并发方法创建并发环境,使用 async/await 调用方法。
韦弦zhy
2022/11/11
3.4K0
Swift 中的 Task
AFNetworking源码探究(一) —— 基本介绍
这里和AFN中UIImageView以及SDWebImage中的调用形式都是一样的。这个和AFN中UIImageView的实现是类似的,也不多说了。
conanma
2021/09/02
2630
Swift 中的 MainActor 使用和主线程调度
MainActor 是Swift 5.5中引入的一个新属性,它是一个全局 actor,提供一个在主线程上执行任务的执行器。在构建应用程序时,在主线程上执行UI更新任务是很重要的,在使用几个后台线程时,这有时会很有挑战性。使用@MainActor属性将帮助你确保你的UI总是在主线程上更新。
韦弦zhy
2022/11/14
3.5K0
Swift 中的 MainActor 使用和主线程调度
AVPlayer初体验之边下边播与视频缓存
上篇文章介绍了AVPlayer的基本播放和解码纹理,本文主要利用AVAssetResourceLoaderDelegate实现AVPlayer的边下边播和缓存机制。 基本原理 AVUrlAsset在请求自定义的URLScheme资源的时候会通过AVAssetResourceLoader实例来进行资源请求。它是AVUrlAsset的属性,声明如下: var resourceLoader: AVAssetResourceLoader { get } 而AVAssetResourceLoader请求的时候会把相关
xferris
2018/06/01
7.3K0
推荐阅读
相关推荐
Swift3.0 Alamofire初级应用
更多 >
领券
社区富文本编辑器全新改版!诚邀体验~
全新交互,全新视觉,新增快捷键、悬浮工具栏、高亮块等功能并同时优化现有功能,全面提升创作效率和体验
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验