由浅到深认识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. 登录阿里云容器镜像服务…...
海外媒体宣发:商务视频推广数字化变革全解析-华媒舍
在当今数字化时代,商务视频推广正迎来一场革命性的数字化变革。本文将为您解析这场变革的核心元素和相关内容。 商务视频推广 商务视频推广是一种通过视频形式来宣传和推广产品、服务或品牌的方法。传统的推广方式主要是通过文字和图片进行,而商务视频推…...
树莓派超全系列教程文档--(61)树莓派摄像头高级使用方法
树莓派摄像头高级使用方法 配置通过调谐文件来调整相机行为 使用多个摄像头安装 libcam 和 rpicam-apps依赖关系开发包 文章来源: http://raspberry.dns8844.cn/documentation 原文网址 配置 大多数用例自动工作,无需更改相机配置。但是,一…...
LeetCode - 394. 字符串解码
题目 394. 字符串解码 - 力扣(LeetCode) 思路 使用两个栈:一个存储重复次数,一个存储字符串 遍历输入字符串: 数字处理:遇到数字时,累积计算重复次数左括号处理:保存当前状态&a…...
Cloudflare 从 Nginx 到 Pingora:性能、效率与安全的全面升级
在互联网的快速发展中,高性能、高效率和高安全性的网络服务成为了各大互联网基础设施提供商的核心追求。Cloudflare 作为全球领先的互联网安全和基础设施公司,近期做出了一个重大技术决策:弃用长期使用的 Nginx,转而采用其内部开发…...
【HTML-16】深入理解HTML中的块元素与行内元素
HTML元素根据其显示特性可以分为两大类:块元素(Block-level Elements)和行内元素(Inline Elements)。理解这两者的区别对于构建良好的网页布局至关重要。本文将全面解析这两种元素的特性、区别以及实际应用场景。 1. 块元素(Block-level Elements) 1.1 基本特性 …...
OPENCV形态学基础之二腐蚀
一.腐蚀的原理 (图1) 数学表达式:dst(x,y) erode(src(x,y)) min(x,y)src(xx,yy) 腐蚀也是图像形态学的基本功能之一,腐蚀跟膨胀属于反向操作,膨胀是把图像图像变大,而腐蚀就是把图像变小。腐蚀后的图像变小变暗淡。 腐蚀…...
深度学习习题2
1.如果增加神经网络的宽度,精确度会增加到一个特定阈值后,便开始降低。造成这一现象的可能原因是什么? A、即使增加卷积核的数量,只有少部分的核会被用作预测 B、当卷积核数量增加时,神经网络的预测能力会降低 C、当卷…...
AI病理诊断七剑下天山,医疗未来触手可及
一、病理诊断困局:刀尖上的医学艺术 1.1 金标准背后的隐痛 病理诊断被誉为"诊断的诊断",医生需通过显微镜观察组织切片,在细胞迷宫中捕捉癌变信号。某省病理质控报告显示,基层医院误诊率达12%-15%,专家会诊…...
七、数据库的完整性
七、数据库的完整性 主要内容 7.1 数据库的完整性概述 7.2 实体完整性 7.3 参照完整性 7.4 用户定义的完整性 7.5 触发器 7.6 SQL Server中数据库完整性的实现 7.7 小结 7.1 数据库的完整性概述 数据库完整性的含义 正确性 指数据的合法性 有效性 指数据是否属于所定…...
计算机基础知识解析:从应用到架构的全面拆解
目录 前言 1、 计算机的应用领域:无处不在的数字助手 2、 计算机的进化史:从算盘到量子计算 3、计算机的分类:不止 “台式机和笔记本” 4、计算机的组件:硬件与软件的协同 4.1 硬件:五大核心部件 4.2 软件&#…...
为什么要创建 Vue 实例
核心原因:Vue 需要一个「控制中心」来驱动整个应用 你可以把 Vue 实例想象成你应用的**「大脑」或「引擎」。它负责协调模板、数据、逻辑和行为,将它们变成一个活的、可交互的应用**。没有这个实例,你的代码只是一堆静态的 HTML、JavaScript 变量和函数,无法「活」起来。 …...
