Java 学习博客_14 入门——集合进阶1

简介: 以黑马程序员全套Java教程网课为主。
graph LR
A[集合]-->C[Collection]
A-->M[Map]
C-->L[List]
C-->S[Set]
L-->AL(ArrayList)
L-->LL(LinkedList)
S-->HS(HashSet)
S-->TS(TreeSet)
M-->HM(HashMap)

其中方形的是接口,圆角的是实现类。

因此先学习 Collection 和 Map 接口后,只需要学习下一级的特有的部分。

Collection

java.util.Collection

Collection 是单例集合的最高级接口。他表示一组对象,这些对象也被称为 Collection 的元素。

无法直接实现。可以用多态的方式实现。

Collection<String> c=new ArrayList<String>();
//add(E e) 方法添加元素
c.add("Hello");
c.add("World");
System.out.println(c);//输出['Hello','World'],因为 ArrayList 中重写了 toString 方法,所以得到的不是一串地址
方法 说明
boolean add(E e) 添加元素,永远返回 true
boolean remove(Object o) 从集合中移除指定的元素,成功移除返回 true
void clear() 清空集合中的元素
boolean contains(Object o) 判断集合中是否存在指定的元素
boolean isEmpty() 判断集合是否为空
int size() 集合长度

快捷键:Alt+7,能看到类的所有信息

迭代器 iterator

java.util.Iterator

Iterator 是一个接口。使用 Iterator 迭代器遍历集合元素。

Iterator<String> it=c.iterator();//通过集合对象中的迭代方法获取对象

追根溯源可以得知,Collection.iterator()方法返回的是一个实现了 Iterator 接口的类:Itr.

所以实际上是通过多态的方式实例化的。

方法 说明
E next() 返回下一个元素(越界访问:NoSuchElementException,被请求的元素不存在)
boolean hasNext() 如果存在下一个元素,返回 true
while(it.hasNext()){
    String s=it.next();//这样更好一些,因为元素可能还要做其他操作
    System.out.println(s);
}

it.next() 像指针一样从0开始遍历。

List

java.util.List

继承自 Collection 类,可以使用其中的方法。

List 是有序、有索引的。相较于集合 Set,List 中的元素可以重复。

List<String> l=new ArrayList<String>();
l.add("Hello");
l.add("World");
System.out.println(l);//按存储的顺序输出,["Hello","World"]
方法 说明
void add(int index,E element) 感觉并不用解释
E remove(int index)
E set(int index,E element)
E get(int index)

越界报错 IndexOutOfException。

另一种遍历集合的方法是 for 循环。

for(int i=0;i<l.size();i++)System.out.println(l.get(i));

并发修改异常

当不允许这样的修改时,会抛出此方法。

如:在使用 iterator 遍历过程中,在集合中添加了新的元素。

iterator 调用 next() 方法时,会先调用 checkForComodification() 方法。

checkForComodification() 会检验 modCount 修改集合的次数和expectedModCount 预期修改集合的次数是否相等。

modCount 来自于 ArrayList 的父类:AbstractList,初始值为0.每次使用 add() 方法时,modCount 就会++。

expectedCount 来自于 Itr 类,且初始值为 expectedCount=modCount.

每次实例化一个 Iterator 类,都会调用其子类 Itr 类,使得 expectedCount 等于当前的 modCount。

然后遍历过程中每次调用 next() 方法时,都会先检查一下expectedCount==modCount,即集合有没有被修改。如果没有修改就可以顺利地返回下一个元素;如果有异常 checkForComodificationException,就会终止运行。

graph TB
C[Collection]-->|获取迭代对象|I[iterator]
C-->|多态|Arr[Arraylist]
I-->|多态|Itr[Itr]
I-->W{"iterator.hasNext()==true"}
Itr-->A(expectedCount=modCount)
W-->|no|en(遍历结束)
W-->|yes|N("iterator.next()")
N-->cf("checkForComodification()")
cf-->cfe{"expectedCount==modCount"}

X-->W
cfe-->|no|Y(checkForComodificationException)
cfe-->|yes|X("集合没有被修改,继续遍历")

但是通过 for 循环遍历集合,在遍历过程中对集合进行修改,不会报异常。

ListInterator

通过 List 中的 listInterator() 方法得到。可以从各个方向遍历,迭代期间可以修改列表,还可以获取列表迭代器的当前位置。

方法 说明
boolean hasNext()
E next()
boolean hasPrevious() 反向遍历用到
E previous() 反向遍历用到
void add(E e) 添加元素,注意是通过 listInterator 添加的,不是通过集合添加的!
List<String> l=new ArrayList<String>();
l.add("Hello");
l.add("World");
l.add("java");
ListInterator<String> li=l.listInterator();//实际上是 listItr 类作为 listInterator 类的子类,通过多态实现
while(li.hasNext()){
    System.out.println(l.next());
}

while(li.hasPrevious()){
    System.out.println(l.previous());
}

while(li.hasNext()){
    String s=l.next();
    if(s.equals("World")){
        li.add("!!!");//特别注意,并不是调用了l的 add() 方法,而是 li 的 add() 方法!
    }
    System.out.println(s);
}

在 listItr 类中的 add() 方法,添加完元素之后,有一句expectedModCount=modCount;重新把实际修改值赋给了预期修改值。因此 next() 方法中判断expectedModCount==modCount,即使添加了新元素也仍然是 true. 不会发生并发修改异常。

增强 for

for(E e:数组或 Collection 的集合)
{
    //在这里使用变量e,把其当做集合元素来用
}

for(int i:arr){
    System.out.println(i);
}

for(int i:list){
    System.out.println(i);
}
//内部是一个 iterator 迭代器,不可以中途修改数据,会引发并发修改异常。

几种遍历方式的选择

单纯的遍历:增强 for 最简单。

需要用到索引:普通 for。

迭代器也要会用。

数据结构

后入先出 FILO。

队列

先入先出 FIFO。

数组

可以根据索引查找,查找效率高;增删效率低。

链表

每个结点存储数据和下一个结点的地址值。最后一个结点地址值为空。

相比数组,增删只需要修改增删处前后结点的地址值,效率更高。

但是查询必须从头开始,效率低。

List 的子类

ArrayList 底层数据结构是数组,LinkedList 底层数据结构是链表。

用法一模一样,查询效率高就用 ArrayList,增删效率高用 LinkedList。

LinkList 特有功能

方法 说明
public void addFirst(E e)
public void addLast(E e)
public E getFirst()
public E getLast()
public void removeFirst()
public void removeLast()
目录
相关文章
|
9天前
|
监控 Java 调度
【Java学习】多线程&JUC万字超详解
本文详细介绍了多线程的概念和三种实现方式,还有一些常见的成员方法,CPU的调动方式,多线程的生命周期,还有线程安全问题,锁和死锁的概念,以及等待唤醒机制,阻塞队列,多线程的六种状态,线程池等
66 6
【Java学习】多线程&JUC万字超详解
|
13天前
|
Java
用JAVA架建List集合为树形结构的代码方法
这段代码定义了一个表示树形结构的 `Node` 类和一个用于构建树形结构的 `TreeController`。`Node` 类包含基本属性如 `id`、`pid`、`name` 和 `type`,以及子节点列表 `children`。`TreeController` 包含初始化节点列表并将其转换为树形结构的方法。通过过滤和分组操作实现树形结构的构建。详情可见:[代码示例链接1](http://www.zidongmutanji.com/zsjx/43551.html),[代码效果参考链接2](https://www.257342.com/sitemap/post.html)。
25 5
|
11天前
|
算法 Java 开发者
Java 编程入门:从零到一的旅程
本文将带领读者开启Java编程之旅,从最基础的语法入手,逐步深入到面向对象的核心概念。通过实例代码演示,我们将一起探索如何定义类和对象、实现继承与多态,并解决常见的编程挑战。无论你是编程新手还是希望巩固基础的开发者,这篇文章都将为你提供有价值的指导和灵感。
|
13天前
|
存储 Java 程序员
Java中的集合框架:从入门到精通
【8月更文挑战第30天】在Java的世界里,集合框架是一块基石,它不仅承载着数据的存储和操作,还体现了面向对象编程的精髓。本篇文章将带你遨游Java集合框架的海洋,从基础概念到高级应用,一步步揭示它的奥秘。你将学会如何选择合适的集合类型,掌握集合的遍历技巧,以及理解集合框架背后的设计哲学。让我们一起探索这个强大工具,解锁数据结构的新视角。
|
11天前
|
Java 程序员 UED
Java 中的异常处理:从入门到精通
【8月更文挑战第31天】在Java编程的世界中,异常处理是保持应用稳定性的重要机制。本文将引导你理解异常的本质,学会如何使用try-catch语句来捕获和处理异常,并探索自定义异常类的魅力。我们将一起深入异常的世界,让你的代码更加健壮和用户友好。
|
11天前
|
Java 数据库连接 开发者
Java中的异常处理:从入门到精通
【8月更文挑战第31天】 在编程世界中,错误和异常就像是不请自来的客人,总是在不经意间打扰我们的程序运行。Java语言通过其异常处理机制,为开发者提供了一套优雅的“待客之道”。本文将带你走进Java异常处理的世界,从基础语法到高级技巧,再到最佳实践,让你的程序在面对意外时,也能从容不迫,优雅应对。
|
12天前
|
存储 算法 Java
Java中的集合框架深度解析与实践
【8月更文挑战第31天】在Java编程的海洋中,集合框架扮演着不可或缺的角色。本文将带你领略Java集合框架的魅力,从理论到实践,深入浅出地探索List、Set和Map等核心接口的使用技巧。我们将通过具体代码示例,展示如何在日常开发中高效运用这些工具,让你的代码更加优雅和高效。无论你是初学者还是有经验的开发者,这篇文章都将为你打开一扇通往Java集合世界的大门。
|
12天前
|
Java 开发者
Java 中的异常处理:从入门到精通
【8月更文挑战第31天】在Java的世界中,异常处理是保持程序健壮性的基石。本文将带你探索Java异常处理的奥秘,从基本的try-catch语句到深入理解自定义异常和最佳实践。你将学会如何优雅地处理错误,确保你的代码不仅能够面对意外情况,还能从中恢复。让我们一起开启这段旅程,掌握让程序更加稳定和可靠的技巧吧!
|
12天前
|
存储 人工智能 Java
JAVA集合
【8月更文挑战第31天】
|
4月前
|
安全 Java
从零开始学习 Java:简单易懂的入门指南之不可变集合、方法引用(二十六)
从零开始学习 Java:简单易懂的入门指南之不可变集合、方法引用(二十六)