第一篇:IteratorProtocol

迭代器(iterator)是一种对象,它能够用来遍历标准模板库容器中的部分或全部元素,每个迭代器对象代表容器中的确定的地址。迭代器修改了常规指针的接口,所谓迭代器是一种概念上的抽象:那些行为上像迭代器的东西都可以叫做迭代器。然而迭代器有很多不同的能力,它可以把抽象容器和通用算法有机的统一起来。

Generator已经被IteratorProtocol取代,但仅仅是名字的修改而已,现定义如下:

public protocol IteratorProtocol {
    associatedtype Element
    
    public mutating func next() -> Self.Element?
}

head first 设计模式中有一章是迭代器模式

class DishesIterator:IteratorProtocol {
    var dishes:[String]
    var idx = -1
    typealias Element = String
    
    init(dishes:[String]) {
        self.dishes = dishes
        self.idx = -1
    }
    
    func next() -> String? {
        idx = idx + 1
        if idx < dishes.count {
            return dishes[idx]
        } else {
            return nil
        }
    }
}

let dIterator = DishesIterator(dishes: ["🐟","🦐"])
dIterator.next()
dIterator.next()
dIterator.next()

这么写有点挫,利用 swift 提供的removeFirst 方法改写

class DishesIterator:IteratorProtocol {
    var dishes:[String]
    typealias Element = String
    
    init(dishes:[String]) {
        self.dishes = dishes
    }
    
    func next() -> String? {
        return dishes.removeFirst()
    }
}

let dIterator = DishesIterator(dishes: ["🐟","🦐"])
dIterator.next()

使用泛型更为通用:

class MyIndexingIterator<T>:IteratorProtocol {
    var content:[T]
    
    init(content:[T]) {
        self.content = content
    }
    
    func next() -> T? {
        return content.removeFirst()
    }
}

let dIterator = MyIndexingIterator(content: ["🐟","🦐"])
dIterator.next()

let iIterator = MyIndexingIterator(content: [1,2])
iIterator.next()

let fIterator = MyIndexingIterator(content: [1.1,2.1])
fIterator.next()

泛型的推断时机是在初始化声明之处,[T] 和传入的元素数组类型绑定,接着又和 next()->T? 中的 T 绑定。

实战

读取文件内容的迭代器

class FileLinesIterator: IteratorProtocol {
    typealias Element = String
    var lines: [String] = []
    
    // 这里你可以用传入filename 来读取content try String(contentsOfFile:  lename) 即可
    init (content: String) {
        let contents = content
        let newLine = CharacterSet.newlines
        lines = contents.components(separatedBy: newLine)
    }
    
    func next() -> Element? {
        guard !lines.isEmpty else{return nil }
        let nextLine = lines.removeFirst()
        return nextLine
    }
}

// 模拟从文件中读取的内容 有换行的
var content = "xsdsdsdsdsdssdsdsdqqqq\nwe你好我得到的\ns设定设定所多"

var iterator = FileLinesIterator(content: content)
iterator.next()

我将定期更新有关此分类的学习笔记。
请关注我的微博:Ninth_Day

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 前言 把《C++ Primer》[https://book.douban.com/subject/25708312...
    尤汐Yogy阅读 13,146评论 1 51
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,955评论 18 399
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,469评论 19 139
  • Java基础常见英语词汇(共70个)['ɔbdʒekt] ['ɔ:rientid]导向的 ...
    今夜子辰阅读 8,605评论 1 34
  • 凝玉低头看看左手上的表,时针已经指向七点一刻,这是她这个礼拜第三天加班了。揉揉眼睛,...
    二缺姐阅读 3,940评论 0 3