Java集合中经典的 5种设计模式,打死也要记住啊!

news2025/1/16 11:13:09

集合

  • 一、 迭代器模式(Iterator Pattern)
  • 二、 工厂模式(Factory Pattern)
  • 三、 装饰器模式(Decorator Pattern)
  • 四、 适配器模式(Adapter Pattern)
  • 五、 组合模式(Composite Pattern)

Java 集合框架中的 List、Set、Map 以及其实现类都使用了多种经典的设计模式

一、 迭代器模式(Iterator Pattern)

所有的集合接口(Collection、List、Set、Queue)和 Map 接口都定义了获取迭代器(Iterator)的方法,如 iterator()listIterator()等。每个具体的集合实现类都提供了自己的迭代器实现,如 ArrayList 有 Itr、LinkedList 有 ListItr、HashMap 有 KeyIteratorValueIterator 等。这些迭代器实现了 Iterator 接口,并对应集合的遍历算法。

ArrayList 为例,它的迭代器 Itr 实现如下:

private class Itr implements Iterator<E> {
    int cursor;       // 下一个要访问的元素索引
    int lastRet = -1; // 最后一次返回的元素索引,方便删除操作
    // ...
}

在这里插入图片描述

二、 工厂模式(Factory Pattern)

Collections 工具类中的静态工厂方法(如 nCopiessingletonListunmodifiableList 等)可以创建各种不同的集合实例。为了方便创建Map.Entry实例,Map接口提供了两个静态工厂方法SimpleEntrySimpleImmutableEntry。这两个方法分别用来创建可变和不可变的Map.Entry实例。

public static <T> List<T> singletonList(T o) {
    return new ImmutableCollections.SetN<T>(o);
}

三、 装饰器模式(Decorator Pattern)

Collections 提供了一些修饰方法,用于对集合进行包装,如 synchronizedCollectionunmodifiableCollection 等。这些方法使用了装饰器模式,在原有集合上添加了同步或只读等额外功能。

public static <T> List<T> synchronizedList(List<T> list) {
    return (list instanceof RandomAccess ?
            new SynchronizedRandomAccessList<>(list) :
            new SynchronizedList<>(list));
}

static class SynchronizedList<E> extends SynchronizedCollection<E> implements List<E> {
    // ...
}

四、 适配器模式(Adapter Pattern)

Arrays.asList 方法可以将数组转换为 List 集合,使用了适配器模式。EnumerationIteratorIteratorEnumeration 则用于在旧集合 Enumeration 和新集合 Iterator 之间进行适配。

public static <T> List<T> asList(T... a) {
    return new ArrayList<>(a);
}

五、 组合模式(Composite Pattern)

在Map接口中,每个Map.Entry对象表示了一个键值对(key-value pair),包含了一个键和一个值。通过这种组合结构,Map可以存储多个键值对,并提供了方便的方式来操作这些数据。在Map接口的实现类(如HashMap、TreeMap)中,每个键值对被表示为一个Map.Entry对象,通过Map的方法可以进行各种操作。组合模式允许将对象组合成树形结构,表示“部分-整体”的关系,在Map.Entry接口中,键和值构成了一个整体,即键值对的Entry,使得操作键值对变得简单方便。

  • Map抽象接口内部定义了一个Entry接口,表示键值对。Entry接口包含了获取键、获取值、设置值、判断相等和获取哈希码等方法。
public interface Map<K, V> {
    // ...其他方法
    
    // 内部接口表示键值对
    interface Entry<K, V> {
        K getKey(); // 获取键
        V getValue(); // 获取值
        V setValue(V value); // 设置值,返回旧的值
        boolean equals(Object o); // 判断是否相等
        int hashCode(); // 获取哈希码
    }
    
    // ...其他方法
}

  • 具体的Map实现类(如HashMap)的 Node 内部类就是一个 Entry 的实现,Node类实现了Map.Entry接口,表示了键值对信息。它包含了键、值和下一个节点的引用等信息
static class Node<K,V> implements Map.Entry<K,V> {
    final int hash;
    final K key;
    V value;
    //下一个节点的引用等信息
    Node<K,V> next; 
    // ...
}

在Java JDK 1.8中,AbstractMap 和 AbstractSet 等抽象类使用了组合模式。AbstractMap 是一个抽象类,它提供了Map接口的基本实现,并使用了组合模式来包装具体的 Map 实现。在 AbstractMap 中,最重要的部分是 entrySet() 方法,该方法返回一个Set<Entry<K,V>>类型的集合,表示 Map 的键值对集合。

public abstract class AbstractMap<K,V> implements Map<K,V> {
    
    // 返回一个包含所有 key-value 映射关系的 Set 集合
    public abstract Set<Entry<K,V>> entrySet();
    
    // 其他实现代码
    
    public static class SimpleEntry<K,V> implements Entry<K,V>, java.io.Serializable{
        private static final long serialVersionUID = -8499721149061103585L;

        private final K key;
        private V value;

        // 构造函数
        
        // 实现 equals()、hashCode() 等方法
    }
}

AbstractMap 中,entrySet() 方法返回的是一个 Set<Entry<K,V>>类型的集合。这里直接使用了 Java 中的 Entry 接口作为键值对的表示,而实际上 Entry 是一个内部接口,定义了对映射项(key-value 对)的操作。

AbstractMap 在内部使用了一个具体的实现类SimpleEntry来表示键值对,但是对外部调用者来说,只需要知道 entrySet() 方法返回的是一个 Set<Entry<K,V>> 集合,而不需要关心具体的实现细节。

通过组合模式的设计,AbstractMap 实现了 Map 接口的基本功能,同时将具体的键值对信息封装在内部类中,加强了封装性和灵活性

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1513585.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

《汇编语言》- 读书笔记 - 第17章-外传之 DOSBox-X 调用 int 13 读写磁盘

《汇编语言》- 读书笔记 - 第17章-外传之 DOSBox-X 调用 int 13 读写磁盘 总结dosbox-x.conf 不完美读取成功写入成功参考资料 总结 DOSBox 中访问 int 13h 始终没反应。网上查了下有人说是没支持&#xff0c;建议使用 DOSBox-X 经过无数遍尝试后&#xff1a; 环境状态Win11…

云计算 3月13号 (OSI 七层模型:物理层、数据链路层、网络层、传输层、会话层、表示层、应用层)

走进网络 1.认识计算机 1.计算机网络是由计算机和通讯构成的&#xff0c;网络研究的是“通信”。 ------1946 世界上第一台计算机 2.终端&#xff1a;只有输入和输出功能&#xff0c;没有计算和处理功能。 3.数据&#xff1a;一串数字&#xff08;二进制数&#xff09;&#…

【el-dialog】解决同一组件使用俩个el-dialog,内容被遮罩层覆盖的问题

如果需要在一个 Dialog 内部嵌套另一个 Dialog或者同一组件有多个Dialog时&#xff0c;需要使用 append-to-body属性 &#xff0c;只要在第二次弹框上面加上属性

第十四届蓝桥杯省赛真题 Java A 组【原卷】

文章目录 发现宝藏【考生须知】试题 A \mathrm{A} A : 特殊日期试题 B: 与或异或试题 C : \mathrm{C}: C: 平均试题 D: 棋盘试题 E : \mathrm{E}: E: 互质数的个数试题 F: 阶乘的和试题 G: 小蓝的旅行计划试题 H: 太阳试题 I: 高塔试题 J \mathrm{J} J : 反异或 01 串 发现…

MongoDB常见面试题总结(一)MongoDB面试题及答案

1. MongoDB的特点&#xff1a; 你能简要介绍一下MongoDB吗&#xff1f;它与关系型数据库的主要区别是什么&#xff1f; MongoDB是一个开源、面向文档的NoSQL数据库&#xff0c;它采用了BSON&#xff08;Binary JSON&#xff09;格式存储数据。以下是MongoDB与关系型数据库的主…

【开发】微服务整合Sentinel

目录 前言 1W&#xff1a;什么是Sentinel&#xff1f; 2W&#xff1a;为什么使用Sentinel&#xff1f; 3W&#xff1a;如何使用Sentinel&#xff1f; 1. 在pom.xml中导入Sentinel依赖坐标 2. 配置控制台 3. 访问API接口的任意端点 流量控制 1. 簇点链路 2. 快速入门…

【HTML】1px边框与1px分割线

对比图 箭头标注的是处理过的 1px分割线 使用transform的scaleY进行缩小 码 <div class"mini-heriz"></div><br><div style"border: solid 1px black; width: 300px;height: 1px;"></div> <style> .mini-heriz {wi…

Java的变量类型详解

目录 局部变量 实例变量 类变量&#xff08;静态变量&#xff09; 参数变量 实例分析 总结 在Java这门静态类型的编程语言中&#xff0c;如何巧妙地使用变量&#xff0c;就像是掌握了一把精准的雕刻刀&#xff0c;能让你在编码的世界里自由地创造。变量在Java中的应用无处…

2024年了,SEO优化是不是已经穷途末路了呢?(川圣SEO)蜘蛛池

baidu搜索&#xff1a;如何联系八爪鱼SEO&#xff1f; baidu搜索&#xff1a;如何联系八爪鱼SEO&#xff1f; baidu搜索&#xff1a;如何联系八爪鱼SEO&#xff1f; 2024年了&#xff0c;SEO优化是不是已经穷途末路了呢&#xff1f;#蜘蛛池SEO SEO优化并没有穷途末路。虽然随…

pcl弧度角度换算:rad2deg,deg2rad

角度弧度换算公式: 代码及结果在:cmath 中cos sin等常用函数的坑(弧度角度换算)-CSDN博客 pcl也有自带的rad2deg,deg2rad: 头文件 #include<pcl/common/angles.h> 代码如下 #include <iostream> #include<pcl/common/angles.h> int main() {vector<…

Linux编程4.3 网络编程-数据封装

1、数据封装 2、Internet协议&#xff08;IP&#xff09; IP的主要目的是为数据输入/输出网络提供基本算法&#xff0c;为高层协议提供无连接的传送服务。这意味着在IP将数据递交给接收站点以前不在传输站点和接收站点之间建立对话&#xff08;虚拟链路&#xff09;。它只是封…

「哈哥赠书活动 - 50期」-『AI赋能写作:AI大模型高效写作一本通』

⭐️ 赠书 - 《AI赋能写作&#xff1a;AI大模型高效写作一本通》 ⭐️ 内容简介 本书以ChatGPT为科技行业带来的颠覆性革新为起点&#xff0c;深入探讨了人工智能大模型如何为我们的创作提供强大支持。本书旨在帮助创作者更好地理解AI的价值&#xff0c;并充分利用其能力提升写…

ubuntu安装开源汇编调试器NASM

安装 安装很简单&#xff0c;直接在终端输入以下命令即可 sudo apt-get install nasm 安装完成后&#xff0c;如果可以查看到nasm的版本号即可视为安装成功 nasm -version 测试 创建汇编文件 创建一个asm文件 vim hello.asm 文件内容如下 section .datahello: db …

《ElementPlus 与 ElementUI 差异集合》el-button 属性 type=“text“ 被删除

差异 element-ui el-button中&#xff0c;属性 type"text" 定义文字按钮&#xff0c;也是链接按钮&#xff1b;element-plus el-button中&#xff0c;改为新增属性 link 并与其它 type 值配合使用&#xff1b; // element-ui <el-button type"text"&g…

网络流量监控软件AnaTraf:优化性能、排除故障的最佳选择

目录 导言 网络流量监控的重要性 AnaTraf网络万用表的功能与优势 网络故障排除与优化网络性能 结论 导言 在当今数字化时代&#xff0c;计算机网络已经成为企业和组织的核心基础设施。然而&#xff0c;网络流量的管理和监控对于确保网络性能的稳定和优化至关重要。本文将介…

商业模式的定义及其成功的四大特点

商业模式&#xff0c;作为企业运营和发展的核心架构&#xff0c;描述了企业如何创造价值、传递价值和获取价值的基本逻辑和方法。简单来说&#xff0c;商业模式就是企业为了实现其市场定位、满足客户需求、实现盈利目标而采取的一系列经营策略和行动的总和。 一个成功的商业模式…

【Linux】进程控制与进程调度

Linux进程介绍 进程的基本概念 Linux是多用户、多任务的操作系统。在这样的环境中&#xff0c;各种计算机资源的分配和管理都是以进程为单位进行的。 Linux操作系统包括三种不同类型的进程&#xff1a; 1&#xff09;交互进程&#xff1a;一种由Shell启动的进程。交互进程既可…

Java项目:基于springboot实现的OA协同办公系统(源码+数据库+毕业论文)

一、项目简介 本项目是一套基于Springbootvue实现的付费自习室系统 包含&#xff1a;项目源码、数据库脚本等&#xff0c;该项目附带全部源码可作为毕设使用。 项目都经过严格调试&#xff0c;eclipse或者idea 确保可以运行&#xff01; 该系统功能完善、界面美观、操作简单、…

matplotlib系统学习记录

日期&#xff1a;2024.03.12 内容&#xff1a;将matplotlib的常用方法做一个记录&#xff0c;方便后续查找。 基本使用 # demo01 from matplotlib import pyplot as plt # 设置图片大小,也就是画布大小 fig plt.figure(figsize(20,8),dpi80)#图片大小&#xff0c;清晰度# 准…

AssetBundle打包与加载

官方文档 参照视频 1.AssetBundle打包 1.1设置资源的命名和后缀 命名只支持小写 1.2创建Editor文件夹&#xff0c;在里面创建编辑器打包AssetBundle的脚本 using UnityEditor; using System.IO;public class CreateAssetBundles {[MenuItem("Assets/Build AssetBun…