文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Swift 基于闭包的类型擦除

2024-12-03 02:57

关注

与许多其他语言相比,使Swift更加安全,更不易出错的原因之一是其先进的(并且在某种程度上是不容忍的)类型系统。这是一种语言功能,有时可能会给人留下深刻的印象,使您的工作效率提高很多,而有时却令人沮丧。

今天,我想重点介绍在 Swift 中处理泛型时可能发生的一种情况,以及我通常如何使用基于闭包的类型擦除技术来解决这种情况。

假设我们要编写一个类,使我们可以通过网络加载模型。由于我们不想为应用程序中的每个模型都复制此类,因此我们选择使其成为泛型类,如下所示:

  1. class ModelLoader { 
  2.     func load(completionHandler: (Result) -> Void) { 
  3.         networkService.loadData(from: T.requestURL) { data in 
  4.             do { 
  5.                 try completionHandler(.success(unbox(data: data))) 
  6.             } catch { 
  7.                 let error = ModelLoadingError.unboxingFailed(error) 
  8.                 completionHandler(.error(error)) 
  9.             } 
  10.         } 
  11.     } 

到目前为止,我们现在有了一个 ModelLoader,它能够加载任何模型(只要它是遵守 Unboxable 协议的),并且能够向我们提供requestURL。但是,我们还希望启用使用此模型加载器的代码易于测试,因此我们将其API提取到一个协议中:

  1. protocol ModelLoading { 
  2.     associatedtype Model 
  3.  
  4.     func load(completionHandler: (Result) -> Void) 

这和依赖注入一起使我们能够轻松地在测试中模拟我们的模型加载API。但这带来了一些复杂性——在每当我们要使用此API时,我们现在都必须将其称为协议 ModelLoading,该协议具有相关的类型要求。这意味着仅引用 ModelLoading 是不够的,因为在没有更多信息的情况下编译器无法推断其关联类型。因此,尝试执行以下操作:

  1. class ViewController: UIViewController { 
  2.     init(modelLoader: ModelLoading) { 
  3.         ... 
  4.     } 

会给我们这个错误:

  1. Protocol 'ModelLoading' can only be used as a generic constraint because it as Self or associated type requirements 

但不用担心,我们可以通过使用泛型轻松摆脱此错误,强制执行符合 Modelloading 的具体类型将由API用户指定,并且它将加载我们期待的模型。像这样:

  1. class ViewController: UIViewController { 
  2.     init(modelLoader: T) where T.Model == MyModel { 
  3.         ... 
  4.     } 

这是有效的,但由于我们还希望在我们的视图控制器中引用我们的模型加载程序,我们需要能够指定属性的类型。 T 只在我们的初始化程序的上下文中知道,因此我们无法定义T类型的属性,除非我们使视图控制器类本身成为泛型 - 这将非常迅速使我们进一步陷入到处都是通用课程的兔子洞中(down into a rabit hole 出自爱丽丝梦游记,意只简单的事情变得越来来复杂和荒谬)。

相反,让我们使用类型擦除,使我们能够保存某种 T 的引用,而无需实际使用其类型。这可以通过创建擦除类型的类,例如 包装类 来完成:

  1. class AnyModelLoader: ModelLoading { 
  2.     typealias CompletionHandler = (Result) -> Void 
  3.  
  4.     private let loadingClosure: (CompletionHandler) -> Void 
  5.  
  6.     init(loader: L) where L.Model == T { 
  7.         loadingClosure = loader.load 
  8.     } 
  9.  
  10.     func load(completionHandler: CompletionHandler) { 
  11.         loadingClosure(completionHandler) 
  12.     } 

以上这种类型擦除技术,其实在 Swift 标准库中也很常用,例如在 AnySequence 类型中。基本上,您将关联值要求的协议包装为泛型类型,然后您可以直接使用它而无需使使用它的类也是泛型的。

我们现在可以更新我们之前的 ViewController,使用 AnyModelloader:

  1. class ViewController: UIViewController { 
  2.     private let modelLoader: AnyModelLoader 
  3.  
  4.     init(modelLoader: T) where T.Model == MyModel { 
  5.         self.modelLoader = AnyModelLoader(loader: modelLoader) 
  6.         super.init(nibName: nil, bundle: nil) 
  7.     } 

好了!我们现在拥有一个面向协议的API,具有易于Mock的特性,且仍然可以在普通类中使用,这归功于类型擦除。

现在,奖励时间的时间。上述技术实际上很好,但它确实涉及一个额外的步骤,为我们的代码增加了一些复杂化。但是,事实证明,我们实际上可以直接在我们的视图控制器中进行基于闭合的类型擦除 ——而不是必须通过 AnyModelloader 类。然后,我们的视图控制器将如下所示:

  1. class ViewController: UIViewController { 
  2.     private let loadModel: ((Result) -> Void) -> Void 
  3.  
  4.     init(modelLoader: T) where T.Model == MyModel { 
  5.         loadModel = modelLoader.load 
  6.         super.init(nibName: nil, bundle: nil) 
  7.     } 

与我们的类型擦除类 AnyModelloader 一样,我们可以参考 load 函数作为闭包的实现,并只需在我们的视图控制器中保存引用。现在,每当我们想要加载模型时,我们只需调用 loadmodel,就像我们的任何其他函数或闭包一样:

  1. override func viewWillAppear(_ animated: Bool) { 
  2.     super.viewWillAppear(animated) 
  3.  
  4.     loadModel { result in 
  5.         switch result { 
  6.         case .success(let model): 
  7.             render(model) 
  8.         case .error(let error): 
  9.             render(error) 
  10.         } 
  11.     } 

就是这样!希望在处理Swift代码中的泛型和协议时,您可以找到上述技术。

本文转载自微信公众号「 Swift社区」,可以通过以下二维码关注。转载本文请联系 Swift社区公众号。

 

来源: Swift社区内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯