当前位置: 首页 > news >正文

详解Java中集合的List接口实现的ArrayList方法 | Set接口实现的HashSet方法

集合的概念

      当我们需要保存一组一样(类型相同)的元素的时候,我们应该使用一个容器来存储,数组就是这样一个容器。

数组的特点:     

      数组是一组数据类型相同的元素集合;

      创建数组时,必须给定长度,而且一旦创建长度不能改变;

      一旦数组装满元素,就需要创建一个新的数组,将元素复制过去;

数组的局限:

        如果装满了,就需要数组复制;

        当我们需要从数组中删除或添加一个元素,需要移动后面的元素;

集合的作用:

       在开发实践中,我们需要一些能够动态增长长度的容器来保存我们的数据,java中为了解决数据存储单一的情况,java中就提供了不同结构的集合类,可以让我们根据不同的场景进行数据存储的选择,如Java中提供了 数组实现的集合,链表实现的集合,哈希结构,树结构等。

分类

单列集合:一次放进去一个值 ( 对象 )

Collection接口:定义了单列集合共有的方法,其子接口Set和List分别定义了存储方式

List:可以有重复元素

Set:不可以有重复元素

双列集合: 键    值

集合API

集合体系概述: Java的集合框架是由很多接口、抽象类、具体类组成的,都位于java.util包中。

List 接口及实现类

概念:List继承了Collection接口,有三个实现的类,分别是:

ArrayList       数组列表

LinkedList     链表列表

Vector            数组列表 (是线程安全的)

1.  ArrayList

         ● 底层有一个数组,可以动态扩展数组长度,并提供了一系列方法操作。

特点: 查询快,在中间增加 / 删除慢

注意:集合容器类中默认可以添加Object类型,但是一般建议一个集合对象只保存同一种类型,若保存多个类型后期处理时涉及类型转换问题;所以我们可以通过泛型来加以控制。

  
         ● 泛型:声明类型时可以自定义参数类型
                      ArraysList  <E>
ArrayList的常用方法:

ArrayList<String> arrayList = new ArrayList();arrayList.add("a");arrayList.add("b");arrayList.add("c");arrayList.add(3,"a");arrayList.remove("a");  //根据内容删除匹配的第一个元素,返回值为booleanarrayList.remove(1);    //删除并返回指定位置的元素arrayList.get(1);         //获取指定位置的元素arrayList.set(1,"X");     //替换并返回指定位置的元素arrayList.clear();        //清空集合中的元素arrayList.isEmpty();      //判断集合元素是否为空arrayList.contains("c");  //判断是否包含指定元素arrayList.size();         //返回集合中元素个数

2.  LinkedList

● 底层是一个链表结构,查询效率低,但增加 / 删除元素快

LinkedList中方法及功能和ArrayList中的方法大致相同,只不过LinkedList中多了关于队列相关的操作方法。

3.  Vector

和ArrayList一样,底层也是数组实现,不同的是Vector的方法默认加了锁,线程是安全的。

List接口集合迭代

List集合遍历方式有四种:

●  1.  for循环遍历          ● 2.  增强for循环遍历   

        //List集合遍历方式1:for循环for (int i = 0; i < arrayList.size(); i++) {if ("a".equals(arrayList.get(i))){arrayList.remove("a");//允许修改集合元素i--;            // 要注意索引的变化与元素位置的移动}}System.out.println(arrayList);System.out.println("----------------");//List集合遍历方式2:增强for循环for (String s:arrayList) {System.out.println(s);   //不能修改集合元素}

 3.  迭代器遍历 (Iterator)

        //List集合遍历方式3:使用迭代器遍历//获得集合对象的迭代器对象Iterator<String> it = arrayList.iterator();while(it.hasNext()){String s = it.next();//获取下一个院系if (s.equals("a")) {it.remove();  //使用迭代器对象删除元素}}System.out.println(arrayList);

4.  ListIterator迭代器:

public static void main(String[] args) {ArrayList<String> arrayList = new ArrayList();arrayList.add("a");arrayList.add("b");arrayList.add("c");arrayList.add("d");/*ListIterator迭代器只能对List接口下的实现遍历ListIterator(index)可以从指定位置开始向前向后遍历*/ListIterator<String> listIterator= arrayList.listIterator(arrayList.size());while(listIterator.hasPrevious()){System.out.println(listIterator.previous());//逆序遍历}}

Set接口

● Set接口也继承了Collection接口,Set中所储存的元素是不重复的,但是是无序的,Set中的元素没有索引。

Set接口的实现类

● HashSet 

● 1. HashSet类中的元素不能重复

 public static void main(String[] args) {HashSet set =new HashSet<>();set.add("a");set.add("a");set.add("b");set.add("c");    //元素是不重复的System.out.println(set);}

● 2. HashSet类中的元素是无序的 

public static void main(String[] args) {HashSet set =new HashSet<>();set.add("c");set.add("s");set.add("x");set.add("d");    //元素是无序的System.out.println(set);set.remove("s");//没有索引,只能根据内容遍历set.iterator();//没有索引,要用迭代器遍历}

 HashSet在添加元素时,是如何判断元素重复的?          *  重点  * 

在底层会先调用hashCode(),注意,Object中的hashCode()返回的是对象的地址,此时并不会调用;此时调用的是类中重写的hashCode(),返回的是根据内容计算的哈希值,遍历时,会用哈希值先比较是否相等,会提高比较的效率;但哈希值会存在问题:内容不同,哈希值相同;这种情况下再调equals比较内容,这样既保证效率又确保安全。

例:

这是错误写法,此时默认调用的是Object类中hashCode( ),返回对象地址

import java.util.HashSet;
import java.util.Objects;public class Student {private String name ;private String num;public Student(String name, String num) {this.name = name;this.num = num;}@Overridepublic String toString() {return "Student{" +"name='" + name + '\'' +", num='" + num + '\'' +'}';}public static void main(String[] args) {HashSet<Student> set = new HashSet<>();Student s1 = new Student("小王1","10001");Student s2 = new Student("小王2","10002");Student s3 = new Student("小王3","10003");Student s4 = new Student("小王1","10001");set.add(s1);set.add(s2);set.add(s3);set.add(s4);System.out.println(set);}}

 正确写法应该是,在Student类中重写hashCode()和equals()

package Demo;import java.util.HashSet;
import java.util.Objects;public class Student {private String name ;private String num;@Overridepublic boolean equals(Object o) {if (this == o) return true;if (o == null || getClass() != o.getClass()) return false;Student student = (Student) o;return num.equals(student.num);}@Overridepublic int hashCode() {return Objects.hash(num);}public Student(String name, String num) {this.name = name;this.num = num;}@Overridepublic String toString() {return "Student{" +"name='" + name + '\'' +", num='" + num + '\'' +'}';}public static void main(String[] args) {HashSet<Student> set = new HashSet<>();Student s1 = new Student("小王1","10001");Student s2 = new Student("小王2","10002");Student s3 = new Student("小王3","10003");Student s4 = new Student("小王1","10001");set.add(s1);set.add(s2);set.add(s3);set.add(s4);System.out.println(set);}}

这样就能避免重复了(此图是s1,s4重复,但只输出s1)

补充:如何快速生成hashCode()和equals()的重写方法?

右键选择Generate,选择equals()and hashCode(),选择重写的属性。

● TreeSet

......敬请期待^_^

相关文章:

  • leetcode 热题 100_找到字符串中所有字母异位词
  • 计算机组成原理-第七章 输入/输出系统【期末复习|考研复习】
  • 企业数字人虚拟形象定制解决方案
  • [SpringCloud] OpenFeign核心架构原理 (一)
  • 『运维备忘录』之 RegEx 正则表达式实例汇总
  • 学习JAVA的第十三天(基础)
  • 芋道项目部署问题指南
  • 常用“树”数据结构
  • 通信棒自动化测试工具
  • uni-app 系统状态栏高度CSS变量--status-bar-height
  • istio pod不启动及访问报RBAC错误问题解决
  • JavaWeb Response:设置响应数据
  • 如何更好的引导大语言模型进行编程的高效开发流程?
  • Kali Linux 2024.1
  • Java多线程注意事项(初级程序员必看)
  • 30天自制操作系统-2
  • 8年软件测试工程师感悟——写给还在迷茫中的朋友
  • Apache Zeppelin在Apache Trafodion上的可视化
  • CentOS 7 修改主机名
  • emacs初体验
  • ES6简单总结(搭配简单的讲解和小案例)
  • JavaScript DOM 10 - 滚动
  • Laravel核心解读--Facades
  • Linux编程学习笔记 | Linux多线程学习[2] - 线程的同步
  • orm2 中文文档 3.1 模型属性
  • Theano - 导数
  • Vue学习第二天
  • 基于HAProxy的高性能缓存服务器nuster
  • 技术胖1-4季视频复习— (看视频笔记)
  • 前端路由实现-history
  • 十年未变!安全,谁之责?(下)
  • 使用common-codec进行md5加密
  • (12)Linux 常见的三种进程状态
  • (C语言)二分查找 超详细
  • (Redis使用系列) SpringBoot中Redis的RedisConfig 二
  • (四)Linux Shell编程——输入输出重定向
  • ..回顾17,展望18
  • .【机器学习】隐马尔可夫模型(Hidden Markov Model,HMM)
  • .NET/C# 编译期能确定的字符串会在字符串暂存池中不会被 GC 垃圾回收掉
  • .NET的数据绑定
  • .NET开源的一个小而快并且功能强大的 Windows 动态桌面软件 - DreamScene2
  • .pub是什么文件_Rust 模块和文件 - 「译」
  • /deep/和 >>>以及 ::v-deep 三者的区别
  • @GetMapping和@RequestMapping的区别
  • [ CTF ]【天格】战队WriteUp- 2022年第三届“网鼎杯”网络安全大赛(青龙组)
  • [ vulhub漏洞复现篇 ] Grafana任意文件读取漏洞CVE-2021-43798
  • [C puzzle book] types
  • [c++] C++多态(虚函数和虚继承)
  • [c++] 什么是平凡类型,标准布局类型,POD类型,聚合体
  • [Java] IDEA Scala环境搭建
  • [js] 正则表达式
  • [LeetCode][LCR190]加密运算——全加器的实现
  • [mysql]错误解决之Failed to start MySQL Server
  • [oeasy]python001_先跑起来_python_三大系统选择_windows_mac_linux
  • [one_demo_16]直接插入排序的demo