由浅到深认识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. 登录阿里云容器镜像服务…...

海外媒体宣发:商务视频推广数字化变革全解析-华媒舍
在当今数字化时代,商务视频推广正迎来一场革命性的数字化变革。本文将为您解析这场变革的核心元素和相关内容。 商务视频推广 商务视频推广是一种通过视频形式来宣传和推广产品、服务或品牌的方法。传统的推广方式主要是通过文字和图片进行,而商务视频推…...

文件服务: 功能介绍
文章目录 一、需求背景二、核心功能1、功能要求2、非工能要求 三、存储方式四、实现方式五、核心技术 一、需求背景 二、核心功能 1、功能要求 文件上传文件预览文件分片上传文件分片合并文件秒传文件断点续传文件下载 2、非工能要求 高扩展性:方便添加新的存储…...

php 快速入门(二)
一、运算符 1.1 算术运算符 运算描述举例加$a$b-减,负数功能$a-$b*乘$a*$b/除$a/$b%取余$a%$b <?php$s1 3;$s2 5;$res1 $s1$s2;$res2 $s1-$s2;$res3 $s1*$s2;$res4 $s1/$s2;$res5 $s1%$s2;echo $res1."<br>";echo "${res2}<b…...

java面试题|(1)多线程如何停止一个线程?
在Java中,停止一个线程的方法通常有以下几种: 使用标志位停止线程: 这是一种常见的做法,即通过设置一个标志位,在线程的执行体中检查这个标志位,当标志位满足某个条件时,退出线程执行。 class M…...

使用阿里CICD流水线打包Java项目到阿里的docker镜像私仓,并自动部署到服务器启动服务
文章目录 使用阿里CICD流水线打包Java项目到阿里的docker镜像私仓,并自动部署到服务器启动服务1、功能原理实现2、将自己的Java项目通过Git上传到阿里的代码仓库中,也可以通过绑定Gitee或者GitHub账号进行导入3、创建自己的阿里云镜像私仓3、进入阿里的C…...

Mybatis的核心配置文件
MyBatis的全局配置文件mybatis-config.xml,配置内容如下: properties(属性)settings(全局配置参数)typeAliases(类型别名)typeHandlers(类型处理器)objectFa…...

四川易点慧电子商务抖音小店:安全可靠,购物新选择
随着互联网的飞速发展,电子商务已成为人们日常生活中不可或缺的一部分。在众多电商平台中,四川易点慧电子商务抖音小店以其安全可靠、服务优质的特点,逐渐赢得了消费者的信赖和好评。 一、平台背景实力雄厚 四川易点慧电子商务有限公司是一家…...

基于数据沙箱与LLM用例自愈的UI自动化测试平台
本期作者 项目参与人员: 顾伊凡、陈钰广、张又中、杨雨浩、樊执政、熊梦园、何璇、谭楠 UI自动化测试能够在一定程度上确保产品质量,尤其在降本提效的大背景下,其重要性愈发凸显。理想情况下,UI自动化测试不仅能够能帮我们规避不少…...

面试算法-117-组合总和 III
题目 找出所有相加之和为 n 的 k 个数的组合,且满足下列条件: 只使用数字1到9每个数字 最多使用一次 返回 所有可能的有效组合的列表 。该列表不能包含相同的组合两次,组合可以以任何顺序返回。 示例 1: 输入: k 3, n 7 输出: [[1,2,4…...

邮件接口与第三方平台的集成的方式有哪些?
邮件接口如何实现高效通信?怎么有效地利用邮件接口? 邮件接口与第三方平台的集成已经成为了企业提升工作效率、优化用户体验的关键环节。那么,邮件接口与第三方平台的集成方式究竟有哪些呢?接下来,AokSend就来探讨一下…...

qrcode插件-生成二维码
安装 yarn add qrcodejs2 --save npm install qrcodejs2 --save 使用 <template><div><div id"qrcodeImg"></div><!-- 创建一个div,并设置id --></div> </template> <script> import QRCode from q…...