作者:Ole Begemann,原文链接,原文日期:2016-09-22
译者:BigbigChai;校对:walkingway;定稿:CMB

本文摘自即将出新版的 Swift 进阶(Advanced Swift)一书中的集合协议(Collection Protocols)章节(稍作修改以适合博客文章)。我和 Chris Eidhof 已经基本完成为本书更新到 Swift 3 的工作,很快可以面世。

Swift 中的集合非常强大,但也很复杂。如果你想实现自定义的集合类型,首先需要了解集合协议的原理。即使只是使用标准库中常见的集合类型,它的工作原理仍然十分值得学习,尤其是它可以帮助你理解编译器打印出来的错误信息。

在本文中,我们想探讨一下集合协议的关联类型。这听起来像是一个晦涩的主题,但我认为想要掌握 Swift 中集合类型的关键:在于对理解关联类型的作用、以及为什么需要它们。

概述

集合有五种关联类型。它们声明如下(实际的代码并不是这样,因为 Index 是在 IndexableBase 中声明的,但你明白我的意思就好):

protocol Collection: Indexable,Sequence {
   associatedtype Iterator: IteratorProtocol = IndexingIterator<Self>
   associatedtype SubSequence: IndexableBase,Sequence = Slice<Self>
   associatedtype Index: Comparable // declared in IndexableBase
   associatedtype Indexdistance: SignedInteger = Int
   associatedtype Indices: IndexableBase,Sequence = DefaultIndices<Self>
   ...
}

前四个关联类型继承自基础协议
Sequence,Indexable 和 IndexableBase 1;集合遵循了以上所有的协议,只是 Index 的约束更加严格、余下的协议赋予了不同的默认值。

注意,除了 Index 以外,集合类型的关联类型都有默认值 — 因此遵守集合协议的类型都只需指定 Index 的类型就可以了。虽然你不必过分在意其他的关联类型,但还是应该大致了解一下。

迭代器 Iterator

遵守 Sequence 协议。Sequence 通过创建迭代器来访问它们的元素。迭代器每次产生一个序列的值,并在遍历该序列时追踪它的迭代状态。

迭代器内部有一个称为 Element 的关联类型。Element 类型指定了迭代器的生成值类型。例如,对于 String.CharacterView 的迭代器而言,Element 的类型是 Character。另外,迭代器也定义了它的 Sequence 的 Element 类型;事实上,我们经常能在方法签名、或者 Sequence 和集合的泛型约束中看到对 Iterator.Element 的引用,就是因为 Element 是 IteratorProtocol 的关联类型。

集合的默认迭代器类型是 IndexingIterator <Self>。这是一个非常简单的封装结构体,它使用集合自身的索引来遍历每个元素。标准库中的大多数集合都使用 IndexingIterator 作为迭代器。我们不需要为自定义的集合更改迭代器类型。

子序列 SubSequence

子序列也遵守 Sequence 协议,但是集合约束更加严格:集合的子序列本身也应该是集合。(我们说“应该”而不是“必须”,因为这种约束在目前的类型系统中无法完全表示。)

在返回初始集合片段的操作中,子序列作为其返回类型:

  • prefix 和 suffix — 取开头或末尾的 n 个元素。

  • dropFirst 和 dropLast — 返回删除开头或末尾 n 个元素后的子序列。

  • 拆分(split) — 以指定的分隔符元素拆分序列,并以数组形式返回。

  • 带有 Range <Index> 参数的 subscript(下标)— 返回指定索引范围内的元素片段。

集合的默认子序列类型是 Slice <Self>,它封装了初始的集合(类似于 IndexingIterator ),并存储该片段在初始集合中的起始索引(startIndex)和结束索引(endindex)。

自定义集合的子序列类型非常有用,特别是当它定义为 Self(即集合的片段与集合本身类型相同)的时候。标准库类型中的例子有 String.CharacterView,这让字符串片段的使用更为方便。而一个反例是 Array,它以 ArraySlice 作为片段类型。

索引 Index

索引表示集合中的位置。每个集合都有两个特殊的索引,startIndex 和 endIndex。 startIndex 指向集合的第一个元素,而 endindex 是集合中最后一个元素之后的索引。索引应该是一个哑值,只存储表明元素位置所需的最少信息量。尤其,索引应该尽可能地减少对集合的引用。集合索引必须是可比较的,这是它唯一的要求。也就是说,索引需要有明确的顺序。

比如数组就是用整数作为索引的,但是整数索引不是对所有数据结构都起作用。我们再以 String.CharacterView 为例,Swift 中的字符是大小可变的;如果你想使用整数索引,你有两个选择:

  1. 用索引表示字符串内部存储的偏移量。这种做法十分有效率;访问一个给定索引的元素的复杂度是 O(1)。但是对于索引范围而言会有差别。例如,如果索引 0 处的字符是正常大小的两倍,则下一个字符的索引会是 2 - 访问索引 1 处的元素将触发致命错误或未定义行为。这会严重违反用户的期望。

  2. 用索引 n 表示字符串中的第 n 个字符。这与用户期望一致 — 对索引范围来说不会有任何差别。然而,访问给定索引的元素的复杂度变成了O(n);必须从头遍历字符串内该索引之前的所有元素,才能确定字符的储存位置。这种行为非常不好,因为用户会期望通过索引下标访问元素的操作能瞬间完成。

因此,String.CharacterView.Index 是一个不可见的值,指向字符串的内部存储缓冲区中的位置。实际上,它只是封装了一个整型偏移量,集合的使用者并不会对这种实现细节感兴趣。
每个集合都需要分别选择正确的索引类型。因此,关联类型中索引是唯一没有默认值的。

索引距离 IndexDistance

索引距离是一个带符号的整型,表示两个索引之间的距离。默认值是整型,我们没必要自己修改。

索引范围 Indices

这是集合的 indices 属性的返回类型。它是一个包含所有索引的集合,该集合中的索引以升序排列对应初始集合的下标。注意,endIndex 不包括在内,因为 endindex 表示”结束之后”的位置,所以不是有效的下标参数。

在 Swift 2 中,indices 属性返回一个 Range <Index>,可以用来遍历集合中所有的有效索引。在 Swift 3 中,Range <Index> 不再可迭代,因为索引不能自我递进(现在由集合来推进索引迭代)。Indices 类型替代了 Range <Index> 来实现索引的迭代。

默认的 Indices 类型十分具有想象力地命名为 DefaultIndices <Self>(23333)。它跟 Slice 一样,是对初始集合、起始和结束索引的一个简单封装 — 它需要保留对初始集合的引用,以便能够推进索引。如果在集合迭代索引的过程中对集合进行修改,可能会导致意想不到的性能问题:假设集合的实现使用了写时复制(copy-on-write)(正如标准库中的所有集合类型),迭代开始之后对集合的额外引用可能触发不必要的复制。

我们在书中广泛说明了写时复制的内容。就现在来说,知道自定义集合可以使用一个不引用初始集合的 Indices 类型就足够了,这样做是一个非常有益的优化。所有索引不依赖于集合本身的集合都可以这样使用,例如数组。如果数组的索引是一个整数类型,你可以使用 CountableRange <Index>。以下是对自定义队列类型的定义(我们在书中实现了此类型):

extension Queue: Collection {
   ...
   
   typealias Indices = CountableRange<Int>
   
   var indices: CountableRange<Int> {
     return startIndex..<endindex
   }
}

本文由 SwiftGG 翻译组翻译,已经获得作者翻译授权,最新文章请访问 http://swift.gg。

  1. 请将 Indexable 和 IndexableBase 视为实现细节。这些协议被引入来解决类型系统不支持递归协议的约束,例如引用了集合本身的集合,它的关联类型就会有这种限制。
    考虑这是更好地支持泛型的部分,希望明年当这个让人高度期待的特性出来时,Indexable 和 IndexableBase 会被去掉,而把它们的功能放在集合内部。 ↩

老司机带你深入浅出 Collection的更多相关文章

  1. ios – 如何在swift3中增加String.Index?

    在swift2.3中运算符用于string.index增加例如.一世我改为swift3代码发生了“一元运算符”不能应用于’@valueString.Index’类型的操作数(又名’@lvalueString.CharacterView.Index’)“在swift3中我改写了例如.i=1但是这段代码无法解决.请帮我.解决方法String.Index是String.CharacterView.Ind

  2. ios – CoreData有序关系 – 使用NSFetchRequest批量取消

    或者,是否存在批量不支持的API,它不是私有的?解决方法目前我有一个解决方案,但不是一个干净的解决方案:我希望按照有序关系中的20个小组进行批量修改.所以,每次我索引一个索引,它的索引除以20,我为接下来的20使用新的NSFetchRequest,并通过调用公共字段名称来解除它们.

  3. ios – Swift中的PageViewController当前页面索引

    我想获取一个pageViewController的当前索引,我不知道我如何获取可见页索引.解决方法您可以使用didFinishAnimating,并将标签设置为查看控制器.尝试这个

  4. ios – OpenGL – 为什么GL_ELEMENT_ARRAY_BUFFER的索引?

    我目前是OpenGLES2.0领域的新手,希望尽可能地了解绑定,缓冲区,着色器等.截至目前,我只是想了解GL_ELEMENT_ARRAY_BUFFER和GL_ARRAY_BUFFER之间的差异,以及何时使用每个注释的预设.我目前的理解使我相信GL_ELEMENT_ARRAY_BUFFER是专门用于所述三角形的索引,而另一个则是其他的.有人可以详细说明为什么,如果这是正确的?GL_ELEMENT_A

  5. 如何恢复索引功能? (Xcode中)

    我的一个项目刚刚开始干扰索引过程.索引过程在中途冻结,然后突然停止,导致SourceKitService崩溃.我根本无法找到错误的代码;因为似乎没有!)–但它无法被索引.最初,我以为它是一个Xcode7.2的问题,所以升级到最新的beta(7.3);但是问题依然存在.我无法恢复到我的旧代码,因为太多的工作将被撤销,我无法发现特定的文件.崩溃报告是here.为了澄清,Xcode本身不会崩溃,只有索引过程.关于如何解决这个问题的任何想法?

  6. ios – Swift:通过索引移动数组中的元素

    给定n个元素的阵列,即vararray=[1,2,3,4,5]我可以写一个扩展到Array,所以我可以修改数组来实现这个输出:[2,5,1]:有没有办法实现这样的功能,可以通过任何索引(正或负)来移动数组.我可以用if-else子句强制执行这个功能,但是我正在寻找的是功能实现.算法很简单:>按提供的索引将数组拆分成两个>将第一个数组追加到第二个数组的末尾有没有什么办法实现它的功能风格?

  7. ios – 从imageview点击手势获取索引或标签值

    这是来自应用商店的图像,只要我们搜索任何应用程序.我也想添加相同的scrollview概念,它显示当前图像与上一个和下一个图像的小预览.我可以在Samplecode的帮助下做出这个观点.但是当用户点击任何图像时,没有找到任何解决方案来获取索引或标签值.所以我可以打开每个图像的详细页面.如果有人有这个想法,请帮助我.提前致谢.解决方法将手势识别器添加到必要的图像视图中:然后在手势处理程序中访问附加到的视图手势识别器:

  8. ios – 不能下标'[NSObject:AnyObject]类型的值?具有“String”类型的索引

    意味着一个可选的类型,这意味着你试图在本质上是一个枚举上调用一个下标.当你尝试这样做时,没有下标声明,所以系统阻塞.通过添加?我们在说,如果可能,打开这个值,然后调用下标.这样一来,系统就会推测出下面的声明类型[NSObject:AnyObject],一切都可以.你也可以使用!强制解开,但如果苹果没有,这将会崩溃.写另一种可能的方式是:这样,苹果不再是可选的,它将始终具有下标语法.不需要解开包装

  9. iOS DeepLinking中是否需要Google App Indexing SDK?

    我想在我的网页和iOS应用中使用GoogleAppIndexing.我确实支持使用ApplesSearch的UniversalLinks(或Googlelingo中的深层链接)并相应地设置我的网页.从Googlesdocumentation开始,我无法确定是否真的需要添加GoogleAppIndexingSDK.SDK没有给我任何必需的功能,我宁愿跳过它–但谷歌是否依靠SDK才能做到这一点?我没有

  10. ios – Swift中的NSDictionary:不能下标“AnyObject”类型的值吗?索引类型为’Int’

    所以我试图使用swift解析JSON中的一些数据.下面是我的代码上面的代码将返回这样的内容然后我尝试使用jsonResult[“subject”]访问所有主题,到目前为止一切顺利但是当我尝试访问个别主题时,例如jsonResult[“subject”][0],Xcode给出了错误:不能下标“AnyObject”类型的值吗?

随机推荐

  1. Swift UITextField,UITextView,UISegmentedControl,UISwitch

    下面我们通过一个demo来简单的实现下这些控件的功能.首先,我们拖将这几个控件拖到storyboard,并关联上相应的属性和动作.如图:关联上属性和动作后,看看实现的代码:

  2. swift UISlider,UIStepper

    我们用两个label来显示slider和stepper的值.再用张图片来显示改变stepper值的效果.首先,这三个控件需要全局变量声明如下然后,我们对所有的控件做个简单的布局:最后,当slider的值改变时,我们用一个label来显示值的变化,同样,用另一个label来显示stepper值的变化,并改变图片的大小:实现效果如下:

  3. preferredFontForTextStyle字体设置之更改

    即:

  4. Swift没有异常处理,遇到功能性错误怎么办?

    本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@foxmail.com举报,一经查实,本站将立刻删除。

  5. 字典实战和UIKit初探

    ios中数组和字典的应用Applicationschedule类别子项类别名称优先级数据包contactsentertainment接触UIKit学习用Swift调用CocoaTouchimportUIKitletcolors=[]varbackView=UIView(frame:CGRectMake(0.0,0.0,320.0,CGFloat(colors.count*50)))backView

  6. swift语言IOS8开发战记21 Core Data2

    上一话中我们简单地介绍了一些coredata的基本知识,这一话我们通过编程来实现coredata的使用。还记得我们在coredata中定义的那个Model么,上面这段代码会加载这个Model。定义完方法之后,我们对coredata的准备都已经完成了。最后强调一点,coredata并不是数据库,它只是一个框架,协助我们进行数据库操作,它并不关心我们把数据存到哪里。

  7. swift语言IOS8开发战记22 Core Data3

    上一话我们定义了与coredata有关的变量和方法,做足了准备工作,这一话我们来试试能不能成功。首先打开上一话中生成的Info类,在其中引用头文件的地方添加一个@objc,不然后面会报错,我也不知道为什么。

  8. swift实战小程序1天气预报

    在有一定swift基础的情况下,让我们来做一些小程序练练手,今天来试试做一个简单地天气预报。然后在btnpressed方法中依旧增加loadWeather方法.在loadWeather方法中加上信息的显示语句:运行一下看看效果,如图:虽然显示出来了,但是我们的text是可编辑状态的,在storyboard中勾选Editable,再次运行:大功告成,而且现在每次单击按钮,就会重新请求天气情况,大家也来试试吧。

  9. 【iOS学习01】swift ? and !  的学习

    如果不初始化就会报错。

  10. swift语言IOS8开发战记23 Core Data4

    接着我们需要把我们的Rest类变成一个被coredata管理的类,点开Rest类,作如下修改:关键字@NSManaged的作用是与实体中对应的属性通信,BinaryData对应的类型是NSData,CoreData没有布尔属性,只能用0和1来区分。进行如下操作,输入类名:建立好之后因为我们之前写的代码有些地方并不适用于coredata,所以编译器会报错,现在来一一解决。

返回
顶部