Swift之深入解析基于闭包的类型擦除

  • 与许多其它语言相比,使 Swift 更加安全,更不易出错的原因之一是其先进的(并且在某种程度上是不容忍的)类型系统,这是一种语言功能,有时可能会给人留下深刻的印象,使我们的工作效率提高很多,而有时却令人沮丧。
  • 在 Swift 中处理泛型时,可能发生的一种情况,以及通常如何使用基于闭包的类型擦除技术来解决这种情况。
  • 假设要编写一个类,可以通过网络加载模型,由于不想为应用程序中的每个模型都复制此类,因此选择使其成为泛型类,如下所示:
class ModelLoader<T: Unboxable & Requestable> {
    func load(completionHandler: (Result<T>) -> Void) {
        networkService.loadData(from: T.requestURL) { data in
            do {
                try completionHandler(.success(unbox(data: data)))
            } catch {
                let error = ModelLoadingError.unboxingFailed(error)
                completionHandler(.error(error))
            }
        }
    }
}
  • 到目前为止,有了一个 ModelLoader,它能够加载任何模型(只要它是遵守 Unboxable 协议的),并且能够向我们提供 requestURL。但是,我们还希望启用使用此模型加载器的代码易于测试,因此将其 API 提取到一个协议中:
protocol ModelLoading {
    associatedtype Model

    func load(completionHandler: (Result<Model>) -> Void)
}
  • 这和依赖注入一起能够轻松地在测试中模拟我们的模型加载 API,但这带来了一些复杂性,在每当要使用此 API 时,都必须将其称为协议 ModelLoading,该协议具有相关的类型要求。这意味着仅引用 ModelLoading 是不够的,因为在没有更多信息的情况下编译器无法推断其关联类型。因此,尝试执行以下操作:
class ViewController: UIViewController {
    init(modelLoader: ModelLoading) {
        ...
    }
}
  • 会提示如下错误:
Protocol 'ModelLoading' can only be used as a generic constraint because it as Self or associated type requirements
  • 但不用担心,我们可以通过使用泛型轻松摆脱此错误,强制执行符合 Modelloading 的具体类型将由 API 用户指定,并且它将加载期待的模型,像这样:
class ViewController: UIViewController {
    init<T: ModelLoading>(modelLoader: T) where T.Model == MyModel {
        ...
    }
}
  • 这是有效的,但由于我们还希望在视图控制器中引用模型加载程序,需要能够指定属性的类型。 T 只在初始化程序的上下文中知道,因此无法定义 T 类型的属性,除非使视图控制器类本身成为泛型。相反,让我们使用类型擦除,能够保存某种 T 的引用,而无需实际使用其类型,这可以通过创建擦除类型的类,例如“包装类”来完成:
class AnyModelLoader<T>: ModelLoading {
    typealias CompletionHandler = (Result<T>) -> Void

    private let loadingClosure: (CompletionHandler) -> Void

    init<L: ModelLoading>(loader: L) where L.Model == T {
        loadingClosure = loader.load
    }

    func load(completionHandler: CompletionHandler) {
        loadingClosure(completionHandler)
    }
}
  • 以上这种类型擦除技术,其实在 Swift 标准库中也很常用,例如在 AnySequence 类型中。基本上,将关联值要求的协议包装为泛型类型,然后可以直接使用它而无需使使用它的类也是泛型的。现在可以更新之前的 ViewController,使用 AnyModelloader:
class ViewController: UIViewController {
    private let modelLoader: AnyModelLoader<MyModel>

    init<T: ModelLoading>(modelLoader: T) where T.Model == MyModel {
        self.modelLoader = AnyModelLoader(loader: modelLoader)
        super.init(nibName: nil, bundle: nil)
    }
}
  • 至此,我们现在拥有一个面向协议的 API,具有易于 Mock 的特性,且仍然可以在普通类中使用,这归功于类型擦除。上述技术实际上很好,但它确实涉及额外的步骤,为代码增加了一些复杂化。但是,事实证明,我们实际上可以直接在视图控制器中进行基于闭合的类型擦除 ,而不是必须通过 AnyModelloader 类。然后视图控制器将如下所示:
class ViewController: UIViewController {
    private let loadModel: ((Result<MyModel>) -> Void) -> Void

    init<T: ModelLoading>(modelLoader: T) where T.Model == MyModel {
        loadModel = modelLoader.load
        super.init(nibName: nil, bundle: nil)
    }
}
  • 与类型擦除类 AnyModelloader 一样,可以参考 load 函数作为闭包的实现,并只需在视图控制器中保存引用。现在,每当我们想要加载模型时,只需调用 loadmodel,就像任何其他函数或闭包一样:
override func viewWillAppear(_ animated: Bool) {
    super.viewWillAppear(animated)

    loadModel { result in
        switch result {
        case .success(let model):
            render(model)
        case .error(let error):
            render(error)
        }
    }
}
  • 4
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
闭包是JavaScript中一个重要且强大的概念。它指的是函数能够访问自己定义范围之外的变量,并且这些变量在函数执行完毕后仍然可以被访问到。闭包的应用场景很广泛,它可以用来创建私有变量、实现模块化、延迟执行等等。了解和掌握闭包对于理解JavaScript语言的本质,以及提高代码的可维护性和灵活性都非常有帮助。 从技术的角度来看,所有的JavaScript函数都可以被看作是闭包。因为在函数内部,它们可以访问在函数定义范围之外的变量。这也是为什么在JavaScript中,函数可以在定义之后仍然可以访问外部的变量。 总结起来,闭包是JavaScript中一个非常重要的概念,它能够让函数访问自己定义范围之外的变量,并且这些变量在函数执行完毕后仍然可以被访问到。了解闭包可以帮助我们更好地理解JavaScript语言的本质,并且提高代码的可维护性和灵活性。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [深入理解JavaScript闭包](https://blog.csdn.net/qq_54384868/article/details/130182961)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *3* [JavaScript深入闭包](https://blog.csdn.net/TZOF_/article/details/117048674)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

╰つ栺尖篴夢ゞ

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值