二十三种设计模式(十六)--迭代器模式

迭代器模式

迭代器模式通过统一的接口解决元素遍历的问题, 屏蔽掉元素复杂的算法细节, 从而让调用者专注于实际业务的处理.

示例代码如下:

java 复制代码
// 具体的元素实现
class Message {
    String title;
    String Content;

    public Message(String title, String content) {
        this.title = title;
        Content = content;
    }
}

class Massager {
    private Message[] msgArr = new Message[5];
    private int size = 0;

    public boolean addMsg(Message msg) {
        if (size < 5) {
            msgArr[size++] = msg;
        } else {
            return false;
        }

        return true;
    }

    // 外部获取迭代器对象的入口
    MessageIterator createIterator() {
        return new MessageIterator();
    }

    // 较为通用的迭代器接口, 也可以再定义其他内容
    public interface iterator {
        Message next();
        boolean hasNext();
    }

    // 迭代器具体实现, 这里采用最简实现方式, 实际根据业务需求确定迭代逻辑
    // 比如采用ArrayList, HashMap等多种方式存储的数据
    // 下一条取用逻辑都在这里封装, 这一部分对调用端隐藏.
    class MessageIterator implements iterator {
        int curIdx = 0;

        @Override
        public Message next() {
            return msgArr[curIdx++];
        }

        @Override
        public boolean hasNext() {
            return curIdx < size;
        }
    }
}

在这个示例中, 迭代器继承自统一的接口, 意味着我们可以从不同的角度封装各种各样的迭代器, 可以有多个迭代器实现
createIterator迭代器创建方法, 在这个示例中仅返回了唯一的迭代器, 实际上根据业务需求, 可以采用参数方式返回多种迭代器

还有一点需要注意的是, 这个示例中, 将迭代器的接口和实现都是采用内部类的方式实现的, 这样也是实际业务中常用的写法, 好处在于迭代器中可以自由访问各种各样的集合容器, 比如示例中的msgArr

如果不采用这种方式, 那么在迭代器创建时要传入容器, 便于迭代器实现遍历逻辑.

外部调用

java 复制代码
public class IteratorPattern {
    public static void main(String[] args) {
        Massager msgr = new Massager();
        msgr.addMsg(new Message("新闻", "吴国近日将伐蜀"));
        msgr.addMsg(new Message("天气预报", "近日东吴有雨"));

        Massager.iterator iterator = msgr.createIterator();
        while (iterator.hasNext()) {
            Message msg = iterator.next();
            System.out.println(msg.title + ": " + msg.Content);
        }
    }
}

迭代器模式实现的必要性不止于此

比如我有如下业务需求:

我有一个名言锦句阅读应用, 客户端不断地从服务器获取未曾阅读过的诗词锦句, 服务端面对大量的用户请求, 进行了三级缓存设计, 以应对大量频繁的病发请求. 针对每一个用户, 首先在Redis中存储10条数据, 当Redis中数据取完时, 会从MongoDB中取用下一组10条数据, MongoDB中则存储100条数据. 当MongoDB中的数据取完时, 再从MySQL数据库中取用下一组100条数据.

而我们具体的业务逻辑中, 只会通过next()方法不断取用下一条数据, 并不关心下一条数据来自哪里.

针对上述诉求, 就可以采用迭代器模式来实现具体的数据取用逻辑

当业务扩展时, 比如我们更换了数据库, 采用PostgreSQL, 不再用MongoDB了, 也只更新迭代器的数据取用逻辑即可, 业务层的next()获取数据逻辑可以做到万年不变.

相关推荐
砚边数影3 小时前
AI数学基础(一):线性代数核心,向量/矩阵运算的Java实现
java·数据库·人工智能·线性代数·矩阵·ai编程·金仓数据库
豆沙沙包?4 小时前
2026年--Lc343-1926. 迷宫中离入口最近的出口(图 - 广度优先搜索)--java版
java·算法·宽度优先
Geoking.4 小时前
【设计模式】23 种设计模式全景总结
设计模式
一颗青果4 小时前
auto | 尾置返回类型 | decltype | using | typedef
java·开发语言·算法
小王不爱笑1324 小时前
SpringBoot 配置文件
java·spring boot·后端
江君是实在人4 小时前
java jvm 调优
java·开发语言·jvm
阿崽meitoufa5 小时前
JVM虚拟机:垃圾收集算法
java·jvm·算法
数电发票API5 小时前
线上充值自动开票攻略:四步落地,告别人工低效内耗
java
想用offer打牌5 小时前
Spring AI vs Spring AI Alibaba
java·人工智能·后端·spring·系统架构
顾北125 小时前
Java接入阿里百炼大模型实战指南
java·ai