首页
学习
活动
专区
圈层
工具
发布
社区首页 >专栏 >【day19】集合和常用API

【day19】集合和常用API

作者头像
程序员波特
发布2024-12-30 08:05:47
发布2024-12-30 08:05:47
2910
举报
文章被收录于专栏:魔法书魔法书

【day18】重点回顾

  1. wait和notify
    • wait: 线程等待,在等待过程中释放锁,需要其他线程调用notify唤醒。
    • notify: 唤醒一条等待的线程,如果有多条线程等待,随机唤醒一条。
    • notifyAll: 唤醒所有等待线程。
  2. Lock: 锁
    • 方法:lock()获取锁,unlock()释放锁。
  3. 线程池: Executors
    • 获取:static ExecutorService newFixedThreadPool(int nThread)
    • 提交线程任务:submit(Runnable r)submit(Callable c)
    • Future: 接收run或者call方法的返回值。
    • shutDown()关闭线程池。
  4. Callable: 类似于Runnable
    • 方法:call()设置线程任务,类似于run方法,但call可以抛出异常,并有返回值。
    • 接收返回值:FutureTask实现了Future接口,get()获取call方法的返回值。

模块19重点

  1. 了解集合的特点及作用。
  2. 掌握Collection接口中的方法。
  3. 掌握使用迭代器迭代集合。
  4. 掌握ArrayListLinkedList的使用。
  5. 掌握使用增强for遍历集合。

第一章:集合框架(单列集合)

在之前的学习中,我们了解了变量和数组用于保存数据,但数组是定长的。当需要添加或删除数据时,数组并不方便,因此我们引入了长度可变的容器——集合。

在这里插入图片描述
在这里插入图片描述
集合的特点
  • 只能存储引用数据类型的数据。
  • 长度可变。
  • 集合中有大量方法,方便操作。
分类
  • 单列集合:一个元素就是一个组成部分,例如list.add("张三")
  • 双列集合:一个元素由两部分构成:keyvalue,例如map.put("涛哥","金莲")

第二章:Collection接口

概述

Collection是单列集合的顶级接口。

使用
  • 创建:Collection<E> 对象名 = new 实现类对象<E>()
  • <E>: 泛型,决定了集合中能存储的数据类型,可以统一元素类型。泛型中只能写引用数据类型,如果不写,默认为Object类型,此时可以存储任何类型的数据。
  • 泛型细节:等号前面的泛型必须写,等号后面的泛型可以不写,JVM会根据前面的泛型推导出后面的泛型。
常用方法
代码语言:javascript
复制
boolean add(E e);  //将给定的元素添加到当前集合中。
boolean addAll(Collection<? extends E> c); // 将另一个集合的元素添加到当前集合中(集合合并)。
void clear(); // 清除集合中所有的元素。
boolean contains(Object o); // 判断当前集合中是否包含指定的元素。
boolean isEmpty(); // 判断当前集合中是否有元素,即判断集合是否为空。
boolean remove(Object o); // 将指定的元素从集合中删除。
int size(); //返回集合中的元素个数。
Object[] toArray(); // 将集合中的元素存储到数组中。
代码语言:javascript
复制
public class Demo01Collection {
    public static void main(String[] args) {
        Collection<String> collection = new ArrayList<>();
        collection.add("萧炎");
        collection.add("萧薰儿");
        collection.add("彩鳞");
        collection.add("小医仙");
        collection.add("云韵");
        collection.add("涛哥");
        System.out.println(collection);
        Collection<String> collection1 = new ArrayList<>();
        collection1.add("张无忌");
        collection1.add("小昭");
        collection1.add("赵敏");
        collection1.add("周芷若");
        collection1.addAll(collection);
        System.out.println(collection1);
        collection1.clear();
        System.out.println(collection1);
        boolean result01 = collection.contains("涛哥");
        System.out.println("result01 = " + result01);
        System.out.println(collection1.isEmpty());
        collection.remove("涛哥");
        System.out.println(collection);
        System.out.println(collection.size());
        Object[] arr = collection.toArray();
        System.out.println(Arrays.toString(arr));
    }
}

第三章:迭代器

1.迭代器基本使用
代码语言:javascript
复制
Iterator<E> iterator(); // 获取迭代器对象。
boolean hasNext(); // 判断集合中是否有下一个元素。
E next(); // 获取下一个元素。
代码语言:javascript
复制
public class Demo01Iterator {
    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<>();
        list.add("楚雨荨");
        list.add("慕容云海");
        list.add("端木磊");
        list.add("上官瑞谦");
        list.add("叶烁");
        Iterator<String> iterator = list.iterator();
        while(iterator.hasNext()){
            String element = iterator.next();
            System.out.println(element);
        }
    }
}

注意:next方法在获取时不要连续使用多次,否则会抛出NoSuchElementException

2.迭代器迭代过程
代码语言:javascript
复制
int cursor; // 下一个元素索引位置
int lastRet = -1; // 上一个元素索引位置
在这里插入图片描述
在这里插入图片描述
3.迭代器底层原理
代码语言:javascript
复制
Iterator iterator = list.iterator();

Iterator是一个接口,等号右边一定是它的实现类对象。

问题:Iterator接收的到底是哪个实现类对象呢?-> ArrayList中的内部类Itr对象。

在这里插入图片描述
在这里插入图片描述

注意:只有ArrayList使用迭代器时Iterator接口才会指向Itr,其他集合使用迭代器Iterator就指向的不是Itr了。

4.并发修改异常

需求:定义一个集合,存储唐僧、孙悟空、猪八戒、沙僧,遍历集合,如果遍历到猪八戒,往集合中添加一个白龙马。

代码语言:javascript
复制
public class Demo03Iterator {
    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<>();
        list.add("唐僧");
        list.add("孙悟空");
        list.add("猪八戒");
        list.add("沙僧");
        Iterator<String> iterator = list.iterator();
        while(iterator.hasNext()){
            String element = iterator.next();
            if ("猪八戒".equals(element)){
                list.add("白龙马");
            }
        }
        System.out.println(list);
    }
}
在这里插入图片描述
在这里插入图片描述

结论:当预期操作次数和实际操作次数不相等时,会出现“并发修改异常”。

我们调用了add方法,而add方法底层只给modCount++,但是再次调用next方法时,并没有给修改后的modCount重新赋值给expectedModCount,导致next方法底层的判断判断出实际操作次数和预期操作次数不相等,所以抛出了“并发修改异常”。

第四章:数据结构

数据结构是一种具有一定逻辑关系,在计算机中应用某种存储结构,并且封装了相应操作的数据元素集合。它包含三方面的内容:逻辑关系、存储关系及操作。

为什么需要数据结构
在这里插入图片描述
在这里插入图片描述

随着应用程序变得越来越复杂和数据越来越丰富,几百万、几十亿甚至几百亿的数据就会出现,而对这么大对数据进行搜索、插入或者排序等的操作就越来越慢,数据结构就是用来解决这些问题的。

数据的逻辑结构指反映数据元素之间的逻辑关系,而与他们在计算机中的存储位置无关:

  • 集合(数学中集合的概念):数据结构中的元素之间除了“同属一个集合”的相互关系外,别无其他关系;
  • 线性结构:数据结构中的元素存在一对一的相互关系;
  • 树形结构:数据结构中的元素存在一对多的相互关系;
  • 图形结构:数据结构中的元素存在多对多的相互关系。
在这里插入图片描述
在这里插入图片描述

数据的物理结构/存储结构:是描述数据具体在内存中的存储(如:顺序结构、链式结构、索引结构、哈希结构)等,一种数据逻辑结构可表示成一种或多种物理存储结构。

数据结构是一门完整并且复杂的课程,那么我们今天只是简单的讨论常见的几种数据结构,让我们对数据结构与算法有一个初步的了解。

1.栈
在这里插入图片描述
在这里插入图片描述
  1. 特点:先进后出。
  2. 好比:手枪压子弹。
2.队列
在这里插入图片描述
在这里插入图片描述
  1. 特点:先进先出。
  2. 好比:过安检。
3.数组
  1. 特点:查询快,增删慢。
  2. 查询快:因为有索引,我们可以直接通过索引操作元素。增删慢:因为数组定长。
    • 添加元素:创建新数组,将老数组中的元素复制到新数组中去,再添加新元素;要是从中间插入就更麻烦了。插入完新元素,后面的元素都要往后移动。
    • 删除元素:创建新数组,将老数组中的元素复制到新数组中去,被删除的元素就不复制了;如果要是从之间删除。被删除的元素后面的元素都要往前移动。
4.链表
在这里插入图片描述
在这里插入图片描述
  1. 在集合中涉及到了两种链表。
  2. 单向链表:
    • 节点:一个节点分为两部分。 第一部分:数据域(存数据)。 第二部分:指针域(保存下一个节点地址)。
    • 特点:前面节点记录后面节点的地址,但是后面节点地址不记录前面节点地址。
  3. 双向链表:
    • 节点:一个节点分为三部分。 第一部分:指针域(保存上一个节点地址)。 第二部分:数据域(保存的数据)。 第三部分:指针域(保存下一个节点地址)。
    • 特点: 前面节点记录后面节点地址,后面节点也记录前面节点地址。
  4. 链表结构特点:查询慢,增删快。
4.1单向链表

a.节点:一个节点分为两部分。 第一部分:数据域(存数据)。 第二部分:指针域(保存下一个节点地址)。 b.特点:前面节点记录后面节点的地址,但是后面节点地址不记录前面节点地址。

4.2双向链表

a.节点:一个节点分为三部分。 第一部分:指针域(保存上一个节点地址)。 第二部分:数据域(保存的数据)。 第三部分:指针域(保存下一个节点地址)。 b.特点: 前面节点记录后面节点地址,后面节点也记录前面节点地址。

在这里插入图片描述
在这里插入图片描述

第五章:List接口

概述

ListCollection接口的子接口。

常见的实现类
  • ArrayList
  • LinkedList
  • Vector

第六章:List集合下的实现类

1.ArrayList集合
代码语言:javascript
复制
1.概述:ArrayList是List接口的实现类。
2.特点:
  a.元素有序。
  b.元素可重复。
  c.有索引,可以利用索引去操作元素。
  d.线程不安全。
3.数据结构:数组。
4.常用方法:
  boolean add(E e) : 将元素添加到集合中。
  void add(int index, E element) : 在指定索引位置上添加元素。
  boolean remove(Object o) : 删除指定的元素。
  E remove(int index) : 删除指定索引位置上的元素,返回的是被删除的那个元素。
  E set(int index, E element) : 将指定索引位置上的元素修改成后面的element元素。
  E get(int index) : 根据索引获取元素。
  int size() : 获取集合元素个数。
代码语言:javascript
复制
public class Demo01ArrayList {
    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<>();
        list.add("铁胆火车侠");
        list.add("喜洋洋");
        list.add("火影忍者");
        list.add("灌篮高手");
        list.add("网球王子");
        System.out.println(list);
        list.add(2,"涛哥");
        System.out.println(list);
        list.remove("涛哥");
        System.out.println(list);
        String element = list.remove(0);
        System.out.println(element);
        System.out.println(list);
        String element2 = list.set(0, "金莲");
        System.out.println(element2);
        System.out.println(list);
        System.out.println(list.get(0));
        System.out.println(list.size());
    }
}
代码语言:javascript
复制
public class Demo02ArrayList {
    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<>();
        list.add("铁胆火车侠");
        list.add("喜洋洋");
        list.add("火影忍者");
        list.add("灌篮高手");
        list.add("网球王子");

        Iterator<String> iterator = list.iterator();
        while(iterator.hasNext()){
            System.out.println(iterator.next());
        }

        System.out.println("=====================");

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

        System.out.println("=====================");
        for (int i = 0; i < list.size(); i++) {
            System.out.println(list.get(i));
        }
    }
}
代码语言:javascript
复制
public class Demo03ArrayList {
    public static void main(String[] args) {
        ArrayList<Integer> list = new ArrayList<>();
        list.add(2);
        System.out.println(list);

        /*
        需求:删除2
        remove(Object o) : 直接删除指定元素。
        remove(int index) : 删除指定索引位置上的元素。

        如果remove中直接传递整数,默认调用按照指定索引删除元素的remove。
        但是此时list中没有2索引,所以越界。

        解决:我们可以将2包装成包装类,变成包装类之后,其父类就是Object了。
        */
        list.remove(Integer.valueOf(2));
        System.out.println(list);
    }
}
1.2.底层源码分析
  1. ArrayList构造方法:
    • ArrayList() 构造一个初始容量为十的空列表。
    • ArrayList(int initialCapacity) 构造具有指定初始容量的空列表。
  2. ArrayList源码总结:
    • 不是一new底层就会创建初始容量为10的空列表,而是第一次add的时候才会创建初始化容量为10的空列表。
    • ArrayList底层是数组,那么为啥还说集合长度可变呢
    • ArrayList底层会自动扩容-> Arrays.copyOf。
    • 扩容多少倍?1.5倍。
代码语言:javascript
复制
ArrayList() 构造一个初始容量为十的空列表
=========================================
private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};
Object[] elementData; ->ArrayList底层的那个数组
    
public ArrayList() {
    this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
}   

=========================================
list.add("a");

public boolean add(E e) {
    modCount++;
    add(e, elementData, size);// e->要存的元素  elementData->集合数组,长度开始为0,size->0
    return true;
}

private void add(E e->元素, Object[] elementData->集合数组, int s->0) {
    if (s == elementData.length)
        elementData = grow();
    elementData[s] = e;
    size = s + 1;
}

private Object[] grow() {
    return grow(size + 1);
}

private Object[] grow(int minCapacity->1) {
    int oldCapacity = elementData.length;//0
    if (oldCapacity > 0 || elementData != DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
        int newCapacity = ArraysSupport.newLength(oldCapacity,
                minCapacity - oldCapacity, /* minimum growth */
                oldCapacity >> 1           /* preferred growth */);
        return elementData = Arrays.copyOf(elementData, newCapacity);
    } else {
        return elementData = new Object[Math.max(DEFAULT_CAPACITY->10, minCapacity->1)];
    }
}
==========================================
// 假设ArrayList中存了第11个元素,会自动扩容-> Arrays.copyOf

private Object[] grow(int minCapacity) {//11
    int oldCapacity = elementData.length;//10
    if (oldCapacity > 0 || elementData != DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
        int newCapacity(15) = ArraysSupport.newLength(oldCapacity->10,
                minCapacity - oldCapacity->1, /* minimum growth */
                oldCapacity >> 1 ->5          /* preferred growth */);
        return elementData = Arrays.copyOf(elementData, newCapacity);
    } else {
        return elementData = new Object[Math.max(DEFAULT_CAPACITY, minCapacity)];
    }
}    


public static int newLength(int oldLength->10, int minGrowth->1, int prefGrowth->5) {
        // preconditions not checked because of inlining
        // assert oldLength >= 0
        // assert minGrowth > 0

        int prefLength = oldLength + Math.max(minGrowth, prefGrowth); // 15
        if (0 < prefLength && prefLength <= SOFT_MAX_ARRAY_LENGTH) {
            return prefLength;
        } else {
            // put code cold in a separate method
            return hugeLength(oldLength, minGrowth);
        }
}
代码语言:javascript
复制
 ArrayList(int initialCapacity);  //构造具有指定初始容量的空列表 
代码语言:javascript
复制
ArrayList<String> list = new ArrayList<>(5);
// ==============================================
public ArrayList(int initialCapacity->5) {
    if (initialCapacity > 0) {
        this.elementData = new Object[initialCapacity];//直接创建长度为5的数组
    } else if (initialCapacity == 0) {
        this.elementData = EMPTY_ELEMENTDATA;
    } else {
        throw new IllegalArgumentException("Illegal Capacity: "+
                                           initialCapacity);
    }
}    
1.1 LinkedList集合
  1. 概述:LinkedList是List接口的实现类。
  2. 特点:
    • 元素有序。
    • 元素可重复。
    • 有索引,这里说的有索引仅仅指的是有操作索引的方法,不代表本质上具有索引
    • 线程不安全。
  3. 数据结构:双向链表。
  4. 方法:有大量直接操作首尾元素的方法。
  • public void addFirst(E e); //将指定元素插入此列表的开头。
  • public void addLast(E e);将指定元素添加到此列表的结尾。
  • public E getFirst();返回此列表的第一个元素。
  • public E getLast();返回此列表的最后一个元素。
  • public E removeFirst();移除并返回此列表的第一个元素。
  • public E removeLast();移除并返回此列表的最后一个元素。
  • public E pop();从此列表所表示的堆栈处弹出一个元素。
  • public void push(E e);将元素推入此列表所表示的堆栈。
  • public boolean isEmpty();如果列表没有元素,则返回true。
代码语言:javascript
复制
public class Demo05LinkedList {
    public static void main(String[] args) {
        LinkedList<String> linkedList = new LinkedList<>();
        linkedList.add("吕布");
        linkedList.add("刘备");
        linkedList.add("关羽");
        linkedList.add("张飞");
        linkedList.add("貂蝉");
        System.out.println(linkedList);

        linkedList.addFirst("孙尚香");
        System.out.println(linkedList);

        linkedList.addLast("董卓");
        System.out.println(linkedList);

        System.out.println(linkedList.getFirst());
        System.out.println(linkedList.getLast());

        linkedList.removeFirst();
        System.out.println(linkedList);

        linkedList.removeLast();
        System.out.println(linkedList);

        System.out.println("======================");

        Iterator<String> iterator = linkedList.iterator();
        while(iterator.hasNext()){
            System.out.println(iterator.next());
        }

        System.out.println("=======================");
        for (int i = 0; i < linkedList.size(); i++) {
            System.out.println(linkedList.get(i));
        }
    }
}

public E pop();从此列表所表示的堆栈处弹出一个元素。 public void push(E e);将元素推入此列表所表示的堆栈。

代码语言:javascript
复制
public class Demo06LinkedList {
    public static void main(String[] args) {
        LinkedList<String> linkedList = new LinkedList<>();
        linkedList.add("吕布");
        linkedList.add("刘备");
        linkedList.add("关羽");
        linkedList.add("张飞");
        linkedList.add("貂蝉");

        linkedList.pop();
        System.out.println(linkedList);
        linkedList.push("涛哥");
        System.out.println(linkedList);
    }
}
1.1 LinkedList底层成员解释说明
  1. LinkedList底层成员
代码语言:javascript
复制
  transient int size = 0;  // 元素个数
  transient Node<E> first;  //第一个节点对象
  transient Node<E> last;  // 最后一个节点对象
  1. Node代表的是节点对象
代码语言:javascript
复制
   private static class Node<E> {
        E item;//节点上的元素
        Node<E> next;//记录着下一个节点地址
        Node<E> prev;//记录着上一个节点地址

        Node(Node<E> prev, E element, Node<E> next) {
            this.item = element;
            this.next = next;
            this.prev = prev;
        }
    }
1.2 LinkedList中add方法源码分析
代码语言:javascript
复制
LinkedList<String> list = new LinkedList<>();
list.add("a");
list.add("b");    

void linkLast(E e) {
        final Node<E> l = last;
        final Node<E> newNode = new Node<>(l, e, null);
        last = newNode;
        if (l == null)
            first = newNode;
        else
            l.next = newNode;
        size++;
        modCount++;
}
1.3.LinkedList中get方法源码分析
代码语言:javascript
复制
public E get(int index) {
    checkElementIndex(index);
    return node(index).item;
} 

Node<E> node(int index) {
    // assert isElementIndex(index);

    if (index < (size >> 1)) {
        Node<E> x = first;
        for (int i = 0; i < index; i++)
            x = x.next;
        return x;
    } else {
        Node<E> x = last;
        for (int i = size - 1; i > index; i--)
            x = x.prev;
        return x;
    }
}

index < (size >> 1)采用二分思想,先将index与长度size的一半比较,如果index<size/2,就只从位置0往后遍历到位置index处,而如果index>size/2,就只从位置size往前遍历到位置index处。这样可以减少一部分不必要的遍历。

第七章:增强for

1.基本使用

1.作用: 遍历集合或者数组。 2.格式:

代码语言:javascript
复制
  for(元素类型 变量名:要遍历的集合名或者数组名){
      变量名就是代表的每一个元素
  }

3.快捷键:集合名或者数组名.for

代码语言:javascript
复制
public class Demo01ForEach {
    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<>();
        list.add("张三");
        list.add("李四");
        list.add("王五");
        list.add("赵六");
        for (String s : list) {
            System.out.println(s);
        }

        System.out.println("=====================");

        int[] arr = {1,2,3,4,5};
        for (int i : arr) {
            System.out.println(i);
        }
    }
}
2.注意
  1. 增强for遍历集合时,底层实现原理为迭代器。
  2. 增强for遍历数组时,底层实现原理为普通for。
在这里插入图片描述
在这里插入图片描述

所以不管是用迭代器还是使用增强for,在遍历集合的过程中都不要随意修改集合长度,否则会出现并发修改异常。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2024-12-29,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 【day18】重点回顾
  • 模块19重点
  • 第一章:集合框架(单列集合)
    • 集合的特点
    • 分类
  • 第二章:Collection接口
    • 概述
    • 使用
    • 常用方法
  • 第三章:迭代器
    • 1.迭代器基本使用
    • 2.迭代器迭代过程
    • 3.迭代器底层原理
    • 4.并发修改异常
  • 第四章:数据结构
    • 为什么需要数据结构
    • 1.栈
    • 2.队列
    • 3.数组
    • 4.链表
    • 4.1单向链表
    • 4.2双向链表
  • 第五章:List接口
    • 概述
    • 常见的实现类
  • 第六章:List集合下的实现类
    • 1.ArrayList集合
    • 1.2.底层源码分析
    • 1.1 LinkedList集合
    • 1.1 LinkedList底层成员解释说明
    • 1.2 LinkedList中add方法源码分析
    • 1.3.LinkedList中get方法源码分析
  • 第七章:增强for
    • 1.基本使用
    • 2.注意
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档