由浅到深认识Java语言(30):集合
该文章Github地址:https://github.com/AntonyCheng/java-notes
在此介绍一下作者开源的SpringBoot项目初始化模板(Github仓库地址:https://github.com/AntonyCheng/spring-boot-init-template & CSDN文章地址:https://blog.csdn.net/AntonyCheng/article/details/136555245),该模板集成了最常见的开发组件,同时基于修改配置文件实现组件的装载,除了这些,模板中还有非常丰富的整合示例,同时单体架构也非常适合SpringBoot框架入门,如果觉得有意义或者有帮助,欢迎Star & Issues & PR!
上一章:由浅到深认识Java语言(29):集合
40.集合
超级for循环
这是JDK1.5出现的特性:循环的特性;
Collection 是单列集合的顶级接口,但是到了 JDK1.5 版本之后,它成为了 Iterable 的子接口;
而只要实现 Iterable 接口,就可以成为超级 for 循环的目标;
所以 Collection List Set 以及数组都实现了这个接口,既能够超级 for 循环
基本格式:
for(数据源中数据的数据类型 迭代变量 : 数据源){System.out.println(迭代变量);
}
示例一:
package top.sharehome.Bag;import java.util.ArrayList;
import java.util.List;public class Demo {public static void main(String[] args) {List list = new ArrayList();list.add("A");list.add("B");list.add("B");list.add("C");list.add("D");list.add("E");list.add("F");for(Object obj:list) {System.out.println(obj);}}
}
打印效果如下:

示例二:
package top.sharehome.Bag;public class Demo {public static void main(String[] args) {int[] arr = {1,2,3,4,5,6};for(int a:arr) {System.out.println(a);}}
}
打印效果如下:

示例三:
package top.sharehome.Bag;public class Demo {public static void main(String[] args) {char[] ch = {'H','e','l','l','o',' ','W','o','r','l','d'};for(char c : ch) {System.out.print(c);}}
}
打印效果如下:

示例四:
package top.sharehome.BigJava;public class Demo {public static void main(String[] args) {int[] arr = new int[]{1, 2, 3, 4, 5, 6, 7, 8, 9};for (int i : arr) {for (int j : arr) {if (j > i) {break;}System.out.print(j + "*" + i + "=" + i * j + " ");}System.out.println();}}
}
打印效果如下:

优缺点
- 优点:能够让我们快速看到一个数据源中具体的元素;
- 缺点:无法对数据源中的数据进行操作;
泛型
出现之前
JDK1.4 之前没有泛型,当时的集合想装什么就可以装什么;
安全机制
软件升级:安全性提高,修复 bug 错误,改善用户体验,增加功能,提升性能;
而 JDK1.5 是Java中的里程碑版本,就是因为加入了泛型的安全机制;
**泛型的作用:**强制了集合存储固定的数据类型,安全性提高了,长远来看程序的代码量会减少,避免了类型的强制转换,程序的问题有运行时问题提前到编译时问题;
初识泛型
集合的特点之一:数据类型是任意的,这个特点既是优点,但也是缺点,因为在实际开发中我们是不会希望集合中的数据类型七七八八,所以我们可以用泛型来约束集合中的数据类型;
在创建集合时指明集合中存放的数据类型,用 <> 包裹起来,此时在集合中,我们就只能引用我们所规定的数据类型,而不能存放其他的数据类型:
package top.sharehome.Bag;import java.util.ArrayList;
import java.util.Collection;public class Demo {public static void main(String[] args) {Collection<String> c = new ArrayList<String>();c.add("a");c.add("B");c.add("c");c.add(92);}
}
报错如下:

使用须知
泛型需要在指明有泛型的类当中使用,例如 Collection 中:

这个 <E> 可以理解成一个**数据类型(可以是基本数据类型,可以是一个对象)**的占位符,当我们改变这个 E 时,该类的方法提示中的数据类型也会跟着改变,但是一定要注意设置数据类型时,基本数据类型必须用包装类,如果我们不设置这个占位符,即删除掉 <E> ,该类会默认填入的数据类型为 Object;
未设置的情况示例:

设置后的情况示例:

注意事项
-
在 Java 1.7 之后可以省略创建对象时声明部分的泛型书写,但要保留创建部分的泛型书写;
但是这里最好是写上!
Collection<Integer> c = new ArrayList<>(); -
一个类中可以创建多个泛型,在使用时也要一并使用,也就是说要么都不指明泛型类型,要么都必须全部指明泛型类型,如果有==一些不明确(表示至少有一种明确的类型)==的类型,可以使用
<已知的一种类型 , Object>来指明;Student 类:
package top.sharehome.Bag; //假设A,B两种类不明确 public class Student<A, B> {private String name;private int age;private A a;private B b;public A getA() {return a;}public void setA(A a) {this.a = a;}public B getB() {return b;}public void setB(B b) {this.b = b;}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;} }Teacher 类:
package top.sharehome.Bag;public class Teacher<String, Integer, Character> {private String name;private int age;private char sex;public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public char getSex() {return sex;}public void setSex(char sex) {this.sex = sex;} }Demo 类:
package top.sharehome.Bag;public class Demo {public static void main(String[] args) {Student stu1 = new Student();Student<String, Object> stu2 = new Student<String, Object>();Teacher T1 = new Teacher();Teacher<String, Integer, Character> T2 = new Teacher<String, Integer, Character>} } -
继承或实现有泛型类型的抽象类或接口;
情况一:在设计子类或者实现类时,直接指明父类的泛型,那么此时父类的泛型就被明确;
Father 类:
package top.sharehome.Bag;public class Father<String> {}Son 类:
package top.sharehome.Bag;public class Son extends Father<String>{}情况二:在设计子类或实现类时,可以为子类设计一个泛型,用于指明父类的泛型;
son 类:
package top.sharehome.Bag;public class Son<T> extends Father<T>{//此时这个T表示的是一个占位符 }Demo 类:
package top.sharehome.Bag;public class Demo {public static void main(String[] args) {Son<Boolean> s = new Son<Boolean>();//此时父类Father中的T就是Boolean} }注意:此时子类中的 T 既指明了父类的泛型,也可以在子类中使用这个 T ;
-
在 Map 集合中使用泛型;
现在就可以很轻松地(不用向下转型)遍历 Map 集合;
示例如下:
package top.sharehome.Bag;import java.util.HashMap; import java.util.Iterator; import java.util.Map; import java.util.Map.Entry; import java.util.Set;public class Demo {public static void main(String[] args) {Map<Integer,String> map = new HashMap<>();map.put(1, "A");map.put(2, "B");map.put(3, "C");Set<Entry<Integer,String>> entrySet = map.entrySet();//使用泛型后,Set集合中存放的元素数据类型就是 Entry 类型System.out.println(entrySet);Iterator<Entry<Integer, String>> iterator = entrySet.iterator();while(iterator.hasNext()) {//这里也可以直接得到Entry,不用向下转型,可以直接输出Entry<Integer,String> next = iterator.next();Integer key = entry.getKey();String value = entry.getValue();System.out.println("entry = " + "\"" + entry + "\"" + " key = " + key + " value = " + value);}} }打印效果如下:

-
自定义泛型类,提高类的可变性;
package top.sharehome.BigJava;public class Demo {public static void main(String[] args) {Student<String> stu = new Student<>();stu.setQ("1911261716");Student<Integer> stu1 = new Student<>();stu1.setQ(1911261716);System.out.println(stu.getQ());System.out.println(stu1.getQ());} }class Student<QQ>{private QQ q;public void setQ(QQ q){this.q = q;}public QQ getQ(){return q;} }打印效果如下:

-
通配符 ?的用法:
package top.sharehome.BigJava;import java.util.ArrayList; import java.util.Iterator; import java.util.List;public class Demo {public static void main(String[] args) {List<String> l = new ArrayList<>();List<Integer> i = new ArrayList<>();l.add("haha");l.add("xixi");l.add("lala");i.add(1);i.add(2);i.add(3);each(l);each(i);}public static void each(List<?> t){Iterator it = t.iterator();while (it.hasNext()){System.out.println(it.next());}} }打印效果如下:

-
泛型的限定:
<? extends Company> 传递类型可以是Company 或者是他的子类;泛型限定限定的是数据类型;
package top.sharehome.BigJava;import java.util.ArrayList; import java.util.Iterator; import java.util.List;public class Demo {public static void main(String[] args) {List<Student> studentList = new ArrayList<Student>();Student stu1 = new Student();stu1.setName("张三");stu1.setAge(21);Student stu2 = new Student();stu2.setName("李四");stu2.setTeamNum(3);studentList.add(stu1);studentList.add(stu2);List<Teacher> teacherList = new ArrayList<Teacher>();Teacher t1 = new Teacher();t1.setName("王五");t1.setAge(39);Teacher t2 = new Teacher();t2.setName("王六");t2.setWorkNum(98);teacherList.add(t1);teacherList.add(t2);work(teacherList);work(studentList);}/*** 此时就明确了该方法参数对象,不会有其他的数据类型或者结构能够传入该方法* @param l*/public static void work(List<? extends Class> l){//这里就是泛型的限定Iterator<? extends Class> it = l.iterator();while (it.hasNext()){Class obj = it.next();System.out.println(obj);}} }class Class {private String name;private int age;public Class(String name, int age) {this.name = name;this.age = age;}public Class() {}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}@Overridepublic String toString() {return "Class{" +"name='" + name + '\'' +", age=" + age +'}';} }class Student extends Class {public Student() {}private int teamNum;public Student(String name, int age, int teamNum) {super(name, age);this.teamNum = teamNum;}public Student(int teamNum) {this.teamNum = teamNum;}public int getTeamNum() {return teamNum;}public void setTeamNum(int teamNum) {this.teamNum = teamNum;}@Overridepublic String toString() {return "Student{" +"teamNum=" + teamNum +"} " + super.toString();} }class Teacher extends Class {public Teacher() {}private int workNum;public Teacher(String name, int age, int workNum) {super(name, age);this.workNum = workNum;}public Teacher(int workNum) {this.workNum = workNum;}public int getWorkNum() {return workNum;}public void setWorkNum(int workNum) {this.workNum = workNum;}@Overridepublic String toString() {return "Teacher{" +"workNum=" + workNum +"} " + super.toString();} }打印效果如下:

注意:泛型上下限的限定;
- <? extends E> 传递 E 类型或者是 E 的子类,这是泛型的上限限定;
- <? super E> 传递 E 类型或者是 E 的父类,这是泛型的下限限定;
相关文章:
由浅到深认识Java语言(30):集合
该文章Github地址:https://github.com/AntonyCheng/java-notes 在此介绍一下作者开源的SpringBoot项目初始化模板(Github仓库地址:https://github.com/AntonyCheng/spring-boot-init-template & CSDN文章地址:https://blog.c…...
Python学习笔记(二)
一:异常: 1.1:异常处理: 1.2:异常捕获: 1.3:异常传递: 二:模块: 2.1:模块的定义: 2.2:模块的导入: 2.3&…...
5.域控服务器都要备份哪些资料?如何备份DNS服务器?如何备份DHCP服务器?如何备份组策略?如何备份服务器状态的备份?
(2.1) NTD(域控数据库)备份 (2.2)DNS备份 (2.3)DHCP备份 (2.4)组策略备份 (2.5)CA证书备份 (2.6)系统状态备份 (2.1)…...
TCP与UDP:网络协议的技术原理与要点
文章目录 1. TCP(传输控制协议)1.1 面向连接1.1.1 三次握手1.1.2 为什么需要三次握手?1.1.3 四次挥手1.1.4 为什么需要四次挥手? 1.2 可靠性1.3 有序传输1.4 流量控制1.5 拥塞控制 2. UDP(用户数据报协议)2…...
vue-office/docx插件实现docx文件预览
1.下包 //预览docx文件 npm install vue-office/docx vue-demi//如果是vue2.6版本或以下还需要额外安装 vue/composition-api2.引入 <template><div>//在src填入文档地址<VueOfficeDocx srchttp://...../xx.docx style"width:80%" rendered"re…...
STM32—控制蜂鸣器(定时器)
目录 1 、 电路构成及原理图 2 、编写实现代码 main.c tim_irq.c 3、代码讲解 4、烧录到开发板调试、验证代码 5、检验效果 此笔记基于朗峰 STM32F103 系列全集成开发板的记录。 1 、 电路构成及原理图 定时器中断是利用定时器的计数功能(向上计数或向下计…...
【React】使用 JSX 为 JavaScript 添加标签
使用 JSX 为 JavaScript 添加标签实际上是将 JSX 语法与 JavaScript 代码结合使用,以描述用户界面。JSX 允许你在 JavaScript 中编写类似 HTML 的结构,并最终由 React 库将其转换为真正的 DOM 元素。以下是将标签引入 JavaScript 以及将 HTML 转化为 JSX…...
Docker构建多平台(x86,arm64)构架镜像
这里写自定义目录标题 背景配置buildx开启experimental重启检查 打包 背景 docker镜像需要支持不同平台架构 配置buildx 开启experimental vi /etc/docker/daemon.json {"experimental": true }或者 重启检查 # 验证buildx版本 docker buildx version# 重启do…...
python爬虫基础-----运算符(第三天)
🎈🎈作者主页: 喔的嘛呀🎈🎈 🎈🎈所属专栏:python爬虫学习🎈🎈 ✨✨谢谢大家捧场,祝屏幕前的小伙伴们每天都有好运相伴左右,一定要天天…...
Itextpdf电子签章
印章 印章是我国特有的历史文化产物,古代主要用作身份凭证和行驶职权的工具。它的起源是由于社会生活的实际需要。早在商周时代,印章就已经产生。如今的印章已成为一种独特的,融实用性和艺术性为一体的艺术瑰宝。传统的印章容易被坏人、小人…...
两台电脑简单的通信过程详解(经过两个路由器,不同网段)
一、eNSP拓扑图 二、配置4台电脑的IP地址、子网掩码、网关地址。 三、配置路由器 注意拓扑图的接口与本博客是否相符,判断以下命令中的ip是否需要修改。 1.AR1-接口对应IP <Huawei>sys #进入系统视图 [Huawei]int g0/0/0 #进入0/0/0接口 [Huawei-GigabitE…...
Java基于微信小程序的助农扶贫系统的研究与实现
博主介绍:✌程序员徐师兄、7年大厂程序员经历。全网粉丝15W、csdn博客专家、掘金/华为云//InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ 🍅文末获取源码联系🍅 👇🏻 精彩专栏推荐订阅👇&#…...
RocketMq方便测试,提供一个controller的接口,支持拉取消息,查看消息内容
通过一个REST API接口动态地启动RocketMQ的消费者,并基于传入的参数(topicName,filterExpression,consumerGroupId)决定要监听哪些消息。在Spring Boot项目中,这通常不是推荐的做法,因为消息消费…...
win10 下Msys2编译FFmpeg的流程方法
安装Msys MSYS2官网 安装... 将\msys64\usr\bin加入环境变量 Mysy2中输入 pacman –Syu pacman –Su pacman -S git patch unzip pacman -S mingw-w64-x86_64-toolchain pacman -S mingw-w64-x86_64-yasm pacman -S mingw-w64-x86_64-SDL2 pacman -S mingw-w64-x86_6…...
用 Delphi 做 FTP 服务器以及如何配置防火墙
使用 Delphi 的 TIdFTPServer 这个控件,做一个 FTP 服务器很简单。可以直接拿官方提供的 FTP Server 的那个 Demo 程序来修改。 本文主要描述如何配置服务器端防火墙 网络环境: 1. 客户端在路由器后面,局域网; 2. 服务器端在路…...
《SQL必知必会第五版》第十四章(组合查询)挑战题
1. 编写 SQL 语句,将两个 SELECT 语句结合起来,以便从 OrderItems表中检索产品 ID(prod_id)和 quantity。其中,一个 SELECT 语句过滤数量为 100 的行,另一个 SELECT 语句过滤 ID 以 BNBG 开头的产品。按产品…...
elasticsearch+kibana安装部分问题:
1.elasticsearch启动问题: 如果elasticsearch开启https登录则第一次启动的时候需要前台启动,前台启动的时候会自己创建相应的token等登录信息,如果是后台启动则没有这些登录信息: ./elasticsearch ━━━━━━━━━━━━━━━━━━━…...
Python---常用的web框架
目录 Django创建Django项目启动Django项目引入APP视图函数例如纯文本JSON格式数据重定向渲染页面返回错误提示 FlaskPyramidTornado Django 特点:Django是一个全功能的Web框架,提供了许多内置的功能和工具,如ORM、表单处理、认证等。它的设计…...
Jenkins Docker 部署指南
Jenkins Docker 部署指南 本文档为您提供了在 Docker 容器中部署 Jenkins 的全面指南,使用的是阿里云的容器镜像服务。请仔细遵循以下步骤以确保成功设置。 前提条件 主机上安装了 Docker。拥有阿里云容器镜像服务的账户。 部署步骤 1. 登录阿里云容器镜像服务…...
海外媒体宣发:商务视频推广数字化变革全解析-华媒舍
在当今数字化时代,商务视频推广正迎来一场革命性的数字化变革。本文将为您解析这场变革的核心元素和相关内容。 商务视频推广 商务视频推广是一种通过视频形式来宣传和推广产品、服务或品牌的方法。传统的推广方式主要是通过文字和图片进行,而商务视频推…...
java调用dll出现unsatisfiedLinkError以及JNA和JNI的区别
UnsatisfiedLinkError 在对接硬件设备中,我们会遇到使用 java 调用 dll文件 的情况,此时大概率出现UnsatisfiedLinkError链接错误,原因可能有如下几种 类名错误包名错误方法名参数错误使用 JNI 协议调用,结果 dll 未实现 JNI 协…...
YSYX学习记录(八)
C语言,练习0: 先创建一个文件夹,我用的是物理机: 安装build-essential 练习1: 我注释掉了 #include <stdio.h> 出现下面错误 在你的文本编辑器中打开ex1文件,随机修改或删除一部分,之后…...
从深圳崛起的“机器之眼”:赴港乐动机器人的万亿赛道赶考路
进入2025年以来,尽管围绕人形机器人、具身智能等机器人赛道的质疑声不断,但全球市场热度依然高涨,入局者持续增加。 以国内市场为例,天眼查专业版数据显示,截至5月底,我国现存在业、存续状态的机器人相关企…...
页面渲染流程与性能优化
页面渲染流程与性能优化详解(完整版) 一、现代浏览器渲染流程(详细说明) 1. 构建DOM树 浏览器接收到HTML文档后,会逐步解析并构建DOM(Document Object Model)树。具体过程如下: (…...
大模型多显卡多服务器并行计算方法与实践指南
一、分布式训练概述 大规模语言模型的训练通常需要分布式计算技术,以解决单机资源不足的问题。分布式训练主要分为两种模式: 数据并行:将数据分片到不同设备,每个设备拥有完整的模型副本 模型并行:将模型分割到不同设备,每个设备处理部分模型计算 现代大模型训练通常结合…...
NFT模式:数字资产确权与链游经济系统构建
NFT模式:数字资产确权与链游经济系统构建 ——从技术架构到可持续生态的范式革命 一、确权技术革新:构建可信数字资产基石 1. 区块链底层架构的进化 跨链互操作协议:基于LayerZero协议实现以太坊、Solana等公链资产互通,通过零知…...
Swagger和OpenApi的前世今生
Swagger与OpenAPI的关系演进是API标准化进程中的重要篇章,二者共同塑造了现代RESTful API的开发范式。 本期就扒一扒其技术演进的关键节点与核心逻辑: 🔄 一、起源与初创期:Swagger的诞生(2010-2014) 核心…...
HDFS分布式存储 zookeeper
hadoop介绍 狭义上hadoop是指apache的一款开源软件 用java语言实现开源框架,允许使用简单的变成模型跨计算机对大型集群进行分布式处理(1.海量的数据存储 2.海量数据的计算)Hadoop核心组件 hdfs(分布式文件存储系统)&a…...
论文笔记——相干体技术在裂缝预测中的应用研究
目录 相关地震知识补充地震数据的认识地震几何属性 相干体算法定义基本原理第一代相干体技术:基于互相关的相干体技术(Correlation)第二代相干体技术:基于相似的相干体技术(Semblance)基于多道相似的相干体…...
虚拟电厂发展三大趋势:市场化、技术主导、车网互联
市场化:从政策驱动到多元盈利 政策全面赋能 2025年4月,国家发改委、能源局发布《关于加快推进虚拟电厂发展的指导意见》,首次明确虚拟电厂为“独立市场主体”,提出硬性目标:2027年全国调节能力≥2000万千瓦࿰…...
