Go设计模式-迭代器到底是不是一个多此一举的模式?

小白debug 2023-06-08 08:00

大家好,这里是陪你一起进步的网管~!上周收到了各位对设计模式的催更,这里我就直接开始啦,今天要一起学习的是迭代器模式。

迭代器模式(Iterator Design Pattern),也叫作游标模式(Cursor Design Pattern)。提供了一种方法顺序地访问一个聚合对象中的元素,而不是暴露该对象的内部表示。这里说的聚合对象也常被称作集合,是编程中最常使用的数据类型之一,有些编程序言还内置提供了整个集合框架,比如Java内置提供的Collection类族、Map类族。

如果集合基于列表, 那么遍历集合这项工作会很简单。但如果集合基于栈、 树、 图和其他复杂的数据结构实现,那么该如何遍历其中的元素呢?

无论集合采用什么数据结构实现, 它都必须提供某种访问元素的方式, 便于其他代码访问其中的元素,怎么才能能让客户端用统一的方式访问不同结构的集合呢?这就需要用到迭代器模式了,迭代器模式的主要思想是将集合的遍历行为抽取为单独的迭代器对象。

迭代器模式用于遍历集合中的对象,很多语言里都内置了这种设计模式,迭代器模式的思想是将集合对象的遍历操作从集合类中拆分出来,放到迭代器实现类中,让两者的职责更单一。

我们可以先看下 Java 语言里内置给我们提供的迭代器,Java中把迭代器定义在java.util.Iterator接口中,接口中的 hasNext 和 next方法是最主要的两个方法,一个用来判断结合中是否还有下个元素,一个用于从集合中取出游标指向的元素,同时把游标指向下个元素。

public interface Iterator<E{

    boolean hasNext();

    next();

    // ... 其他方法省略了
}

不管用哪种语言实现迭代器模式,这两个方法都是必不可少的。

有了迭代器接口 Iterator 后,每种集合都会提供Iterator的具体实现,用来让客户端通过迭代器遍历它们

List list = new ArrayList<>();
list.add("one");
list.add("two");
list.add("three");

Iterator iterator = list.iterator();


Set set = new HashSet<>();
set.add("one");
set.add("two");
set.add("three");

Iterator iterator2 = set.iterator();

while(iterator.hasNext()) {
    Object nextObject = iterator.next();

}

下面我们看一下迭代器模式的构成,在 Java 语言里通过自身的集合框架和Iterator接口实现了迭代器模式的接口抽象和实现,所以写代码时可能不会太关注,待我们了解清楚迭代器模式的结构构成后再用代码自己实现起来也就不那么难了。

迭代器模式的结构

迭代器模式的结构,可以用下面这个 UML 类图表示:

  • Iterator 接口:这个接口会定义一些基础的操作函数,如hasNext()getNext()等。通过名称就可以看出,这些方法可以帮助我们执行遍历集合、重启迭代等操作。
  • Collection 接口:这个接口代表了要被遍历的集合。在这个接口里定义了一个createIterator方法,该方法会返回一个Iterator的实例。
  • Concrete Iterator: Iterator接口的具体实现类。
  • Concrete Collection: Collection接口的具体实现类。
  • 客户端 Client:通过集合和迭代器的接口与两者进行交互。这样一来客户端无需与具体类进行耦合, 允许同一客户端代码使用各种不同的集合和迭代器。

Go 源码实现

迭代器模式的编码实现环节,我们选择用Go语言给底层数据结构为列表的集合实现一个迭代器模式。例子为了演示方便易懂,选择了列表这个看似实现迭代器完全多此一举的集合给大家演示,大家不要忘了咱们上边强调过的,迭代模式的最大收益是让使用者即不必关心集合的底层结构,还能用统一的方式访问这些不同结构的集合。

首先按照迭代器模式的结构我们来声明迭代器和集合的接口。迭代器接口中定义两个核心方法hasNextgetNext

type iterator interface {
 hasNext() bool
 getNext() *User
}

集合要能创建出遍历自身元素的迭代器,所以接口中必须要包含createIterator 方法。

type collection interface {
 createIterator() iterator
}

定义好接口后接下来我们要的是编写集合的具体实现,实现中将集合对象的遍历操作从集合类中拆分出来,放到迭代器实现类中。

"本文使用的完整可运行源码
去公众号「网管叨bi叨」发送【设计模式】即可领取"

type User struct {
 name string
 age int
}

type userCollection struct {
 users []*User
}

func (u *userCollection) createIterator() iterator {
 return &userIterator{
  users: u.users,
 }
}

type userIterator struct {
 index int
 users []*User
}

func (ui *userIterator) hasNext() bool {
 return ui.index < len(ui.users)
}

func (ui *userIterator) getNext() *User {
 if ui.hasNext() {
  user := ui.users[ui.index]
  ui.index++
  return user
 }

 return nil
}

有了迭代器后,客户端就能用迭代器完成集合对象的元素遍历了。

"本文使用的完整可运行源码
去公众号「网管叨bi叨」发送【设计模式】即可领取"

func main() {
 userK := &User{
  name: "Kevin",
  age: 30,
 }
 userD := &User{
  name: "Diamond",
  age:  25,
 }

 userCollection := &userCollection{
  users: []*User{userK, userD},
 }

 iterator := userCollection.createIterator()
 for iterator.hasNext() {
  user := iterator.getNext()
  fmt.Printf("User is %v\n", user)
 }
}

当然上面实现的迭代器遍历代码中没有加锁,所以不是线程安全的,大家可以自己完善下。

本文的完整源码,已经同步收录到我整理的电子教程里啦,可向我的公众号「网管叨bi叨」发送关键字【设计模式】领取。

公众号「网管叨bi叨」发送关键字【设计模式】领取。

总结

迭代器模式在平时编程的时候使用的并不多,像Java、C#编程时都自带了迭代器模式的实现,也支持实现语言内置的Iterator接口来给自定义集合创建迭代器。

关于迭代器的使用场景,我们只要记住它的主要思想是将集合对象的遍历操作从集合类中拆分出来,放到迭代器实现类中,让两者的职责更单一的同时也让客户端不必关系该怎么去实现集合的迭代算法。

个人认为Java语言的Collection、Map类族中提供的各种迭代器是对该模式的典型应用,代码实现写的很优秀,值得借鉴学习。

这里再跟大家安利下,去年我为了多赚点稿费,背着大家偷偷签约掘金平台更新的零基础学Java系列,里面有差不多三篇文章都是介绍集合框架还有迭代器这些内容的,我觉得写的还挺不错的,简单易读,脉络清晰,大家感兴趣的可以去瞅瞅:https://juejin.cn/column/7081836723935641607

好了,本次文章就到这里啦,设计模式还剩五篇,大家后面继续期待吧。

- END -


扫码关注公众号「网管叨bi叨」

给网管个星标,第一时间吸我的知识 👆

网管整理了一本《Go 开发参考书》收集了70多条开发实践。去公众号回复【gocookbook】领取!还有一本《k8s 入门实践》讲解了常用软件在K8s上的部署过程,公众号回复【k8s】即可领取!


觉得有用就点个在看  👇👇👇

评论 (0)
  • 某国产固态电解的2次和3次谐波失真相当好,值得一试。(仅供参考)现在国产固态电解的性能跟上来了,值得一试。当然不是随便搞低端的那种。电容器对音质的影响_电子基础-面包板社区  https://mbb.eet-china.com/forum/topic/150182_1_1.html (右键复制链接打开)电容器对音质的影响相当大。电容器在音频系统中的角色不可忽视,它们能够调整系统增益、提供合适的偏置、抑制电源噪声并隔离直流成分。然而,在便携式设备中,由于空间、成本的限
    bruce小肥羊 2025-05-04 18:14 231浏览
  • 二位半 5线数码管的驱动方法这个2位半的7段数码管只用5个管脚驱动。如果用常规的7段+共阳/阴则需要用10个管脚。如果把每个段看成独立的灯。5个管脚来点亮,任选其中一个作为COM端时,另外4条线可以单独各控制一个灯。所以实际上最多能驱动5*4 = 20个段。但是这里会有一个小问题。如果想点亮B1,可以让第3条线(P3)置高,P4 置低,其它阳极连P3的灯对应阴极P2 P1都应置高,此时会发现C1也会点亮。实际操作时,可以把COM端线P3设置为PP输出,其它线为OD输出。就可以单独控制了。实际的驱
    southcreek 2025-05-07 15:06 184浏览
  • ‌一、高斯计的正确选择‌1、‌明确测量需求‌‌磁场类型‌:区分直流或交流磁场,选择对应仪器(如交流高斯计需支持交变磁场测量)。‌量程范围‌:根据被测磁场强度选择覆盖范围,例如地球磁场(0.3–0.5 G)或工业磁体(数百至数千高斯)。‌精度与分辨率‌:高精度场景(如科研)需选择误差低于1%的仪器,分辨率需匹配微小磁场变化检测需求。2、‌仪器类型选择‌‌手持式‌:便携性强,适合现场快速检测;‌台式‌:精度更高,适用于实验室或工业环境。‌探头类型‌:‌横向/轴向探头‌:根据磁场方向选择,轴向探头适合
    锦正茂科技 2025-05-06 11:36 376浏览
  • UNISOC Miracle Gaming奇迹手游引擎亮点:• 高帧稳帧:支持《王者荣耀》等主流手游90帧高画质模式,连续丢帧率最高降低85%;• 丝滑操控:游戏冷启动速度提升50%,《和平精英》开镜开枪操作延迟降低80%;• 极速网络:专属游戏网络引擎,使《王者荣耀》平均延迟降低80%;• 智感语音:与腾讯GVoice联合,弱网环境仍能保持清晰通话;• 超高画质:游戏画质增强、超级HDR画质、游戏超分技术,优化游戏视效。全球手游市场规模日益壮大,游戏玩家对极致体验的追求愈发苛刻。紫光展锐全新U
    紫光展锐 2025-05-07 17:07 170浏览
  • 想不到短短几年时间,华为就从“技术封锁”的持久战中突围,成功将“被卡脖子”困境扭转为科技主权的主动争夺战。众所周知,前几年技术霸权国家突然对华为发难,导致芯片供应链被强行掐断,海外市场阵地接连失守,恶意舆论如汹涌潮水,让其瞬间陷入了前所未有的困境。而最近财报显示,华为已经渡过危险期,甚至开始反击。2024年财报数据显示,华为实现全球销售收入8621亿元人民币,净利润626亿元人民币;经营活动现金流为884.17亿元,同比增长26.7%。对比来看,2024年营收同比增长22.42%,2023年为7
    用户1742991715177 2025-05-02 18:40 209浏览
  • 文/郭楚妤编辑/cc孙聪颖‍相较于一众措辞谨慎、毫无掌舵者个人风格的上市公司财报,利亚德的财报显得尤为另类。利亚德光电集团成立于1995年,是一家以LED显示、液晶显示产品设计、生产、销售及服务为主业的高新技术企业。自2016年年报起,无论业绩优劣,董事长李军每年都会在财报末尾附上一首七言打油诗,抒发其对公司当年业绩的感悟。从“三年翻番顺大势”“智能显示我第一”“披荆斩棘幸从容”等词句中,不难窥见李军的雄心壮志。2012年,利亚德(300296.SZ)在深交所创业板上市。成立以来,该公司在细分领
    华尔街科技眼 2025-05-07 19:25 117浏览
  • 多功能电锅长什么样子,主视图如下图所示。侧视图如下图所示。型号JZ-18A,额定功率600W,额定电压220V,产自潮州市潮安区彩塘镇精致电子配件厂,铭牌如下图所示。有两颗螺丝固定底盖,找到合适的工具,拆开底盖如下图所示。可见和大部分市场的加热锅一样的工作原理,手绘原理图,根据原理图进一步理解和分析。F1为保险,250V/10A,185℃,CPGXLD 250V10A TF185℃ RY 是一款温度保险丝,额定电压是250V,额定电流是10A,动作温度是185℃。CPGXLD是温度保险丝电器元件
    liweicheng 2025-05-05 18:36 253浏览
  • 2024年初,OpenAI公布的Sora AI视频生成模型,震撼了国产大模型行业。随后国产厂商集体发力视频大模型,快手发布视频生成大模型可灵,字节跳动发布豆包视频生成模型,正式打响了国内AI视频生成领域第一枪。众多企业匆忙入局,只为在这片新兴市场中抢占先机,却往往忽视了技术成熟度与应用规范的打磨。以社交平台上泛滥的 AI 伪造视频为例,全红婵家人被恶意仿冒博流量卖货,明星们也纷纷中招,刘晓庆、张馨予等均曾反馈有人在视频号上通过AI生成视频假冒她。这些伪造视频不仅严重侵犯他人权
    用户1742991715177 2025-05-05 23:08 78浏览
  • 随着智能驾驶时代到来,汽车正转变为移动计算平台。车载AI技术对存储器提出新挑战:既要高性能,又需低功耗和车规级可靠性。贞光科技代理的紫光国芯车规级LPDDR4存储器,以其卓越性能成为国产芯片产业链中的关键一环,为智能汽车提供坚实的"记忆力"支持。作为官方授权代理商,贞光科技通过专业技术团队和完善供应链,让这款国产存储器更好地服务国内汽车厂商。本文将探讨车载AI算力需求现状及贞光科技如何通过紫光国芯LPDDR4产品满足市场需求。 车载AI算力需求激增的背景与挑战智能驾驶推动算力需求爆发式
    贞光科技 2025-05-07 16:54 128浏览
  • 5小时自学修好BIOS卡住问题  更换硬盘故障现象:f2、f12均失效,只有ESC和开关机键可用。错误页面:经过AI的故障截图询问,确定是机体内灰尘太多,和硬盘损坏造成,开机卡在BIOS。经过亲手拆螺丝和壳体、排线,跟换了新的2.5寸硬盘,故障排除。理论依据:以下是针对“5小时自学修好BIOS卡住问题+更换硬盘”的综合性解决方案,结合硬件操作和BIOS设置调整,分步骤说明:一、判断BIOS卡住的原因1. 初步排查     拔掉多余硬件:断开所有外接设备(如
    丙丁先生 2025-05-04 09:14 118浏览
我要评论
0
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦