java中Collection集合关键点记录

本文详细介绍了Java集合框架中的ArrayList、Vector和HashSet的特性和使用,包括它们的底层实现、增删查改操作、线程安全性和效率。此外,还深入探讨了泛型的概念、作用、应用场合以及泛型类和泛型方法的细节,解释了泛型在多态、类型安全和代码简洁性方面的优势。最后,提到了Set接口的特点以及LinkedList和TreeSet的使用和遍历方式。

java中集合总结与理解

写在前面

*数组,集合都是对多个数据进行存储操作,简称为容器

注:这里的存储指的是内存层面的存储,不是磁盘上的持久化存储。

一.集合分类及应用场合

主要是要将一类相同属性的信息整合起来,需要用到集合。集合中可以放多种类型的数据,不受限制。在这里插入图片描述

二.Collection类

1.集合一个特点:

只能存放引用数据类型的数据,如果传入基本数据类型,会进行自动装箱处理。

2.遍历方式

//集合遍历
//1.增强for循环
for(Object o:col){
    System.out.print(o+",");
}
//2.迭代器集合对象拥有iterator方法
Iterator it=col.iterator();
while (it.hasNext()){
    System.out.print(it.next()+",");
}

3.List接口

//List重写方法add()
list.add(1,"avv");
//set修改方法0下标和修改后值
list.set(0,22);
//list.remove(n)n为Intager类型,则删除的是下标n的值
//当n为Obj类型的时,删除的是元素值为n的数据。
list.remove(2);
//get(n)通过索引,输出下标为n的元素值
list.get(1);
//list因为有get方法,所以可以普通for循环遍历
3.1ArrayList实现类
  • jdk1.7–底层源码是数组,调用构造器时,初始化== 数组长度10 ==,扩容的时候为原数组的1.5倍。(有些类似于StringBuilder)
  • jdk1.8–底层源码是数组,调用构造器时,初始化== 数组长度为空 ==,调用add方法后,才给新数组赋长度为10,这样节省了空间。
  • ArrayList线程安全,但效率高
3.2Vector实现类(主要对比与ArrayList的区别)
  • 底层仍然是Object数组,int类型属性表示数组中有效长度
  • 底层数组调用构造器时,初始化== 数组长度10 ==
  • 当调用add方法时,底层数组扩容为原数组的2倍
  • 相对于ArrayList来说,Vector是线程安全的,但效率低下。
3.3 ArrayList和Vector共同特点
  • 由于底层实现是数组实现的,所以查询效率高并且可重复
  • 缺点在于增,删效率低下
3.4泛型
3.4.1作用:
  • 将集合内的数据类型统一化,不是泛型指定类型则不能加入集合,这样会使集合后续遍历简单化.

注:泛型对应数据类型都是引用数据类型

3.4.2实例:
ArrayList<Integer> al=new ArrayList<Integer>();//JDK1.5
ArrayList<INteger> al=new ArrayList<>();  //JDK1.7
3.4.3应用场合

泛型其本质还是多态,在抽象接口或者类的情况下使用泛型先预留好位置,当具体实现类定义集合时,再去真正定义集合的具体类型。

//没有指定泛型的引用数据类型,E可代表任意类型
public class Test003<E> {
    int ads;
    E sex;
    public void a(E n){
    }
    public void b(E[] m){
    }
}
//这时子类E为Integer型,创建对象默认Integer型
class Testson extends Test003<Integer>{

}
//父类不指定泛型类型,子类就也是泛型类,创建对象时才确定E的类型
class Testson01<E> extends Test003<E>{

}

class sds{
    public static void main(String[] args) {
        //首先需要对泛型类实例化对象
        //1.实例化泛型类但不指定泛型,默认为Object类型
        Test003 ts=new Test003();
        ts.a("abc");
        ts.b(new String[]{"s","d","g"});
        //2.实例化明确指定泛型类,E的类型为传入的泛型类(推荐)
        Test003<Integer> ts2=new Test003<>();
        ts2.a(4);

        //继承父类,子类自动是父类的泛型类
        Testson tso=new Testson();
        //自动Integer类
        tso.a(5);
    }
}
3.4.4泛型类的一些特点
  • 泛型类可以定义多个参数类型
  • 泛型类的构造器不能加入泛型
  • 不同泛型的引用类型不可以相互赋值
  • 泛型不指定,被擦除,相当于还是原来的Object类型
  • 泛型类中静态方法不能使用累的泛型(与构造方法原理类似)
  • 不能直接使用E[]进行创建,因为并不知道它是什么类型的
3.4.5泛型方法
  • 要求这个方法的泛型的参数类型和当前类的泛型无关

    //不是泛型方法
    public void a(E e){
    }
    //错误方法
    public static void a(E e){
    }
    //泛型方法
    public<T> void b(T t){
    }
    //泛型静态方法
    public static<T> void b(T t){
    }
    
  • T的类型是在调用该泛型方法的时候确定的,而不是创建类对象时

  • 泛型方法可以是静态方法

问题,为什么泛型方法可以是静态方法,而泛型类方法不能是不能是静态方法?

答:因为普通类需要的E是要在对象创建的时候指定的,而静态方法是随着类就已经产生了,这就出现了先有类,但是没有对象,不知道E的类型; 但是静态泛型方法虽然也是和类一起创建的,但是由于该方法不和类对象绑定,所以即使先有类也不需要提前知道T的类型,调用方法再指定就可以了。

注:A和B是父子类的关系,但是G和G不存在继承关系,是并列的关系。

3.4.6通配符
List<Object> list1=new ArrayList<>();
List<String> list2=new ArrayList<>();
//A和B是父子类的关系,但是G<A>和G<B>不存在继承关系,是并列的关系。
//加入通配符“?”后,G<?>就变成了G<A>和G<B>的父类
List<?> list=null;
list=list1;
list=list2;
  • 通配符作用:当一个方法使用通配符后,可以传入多种类型数据

    注:方法的遍历用增强for循环时,前面应该是Object类型

3.4.7 泛型受限

在这里插入图片描述

3.5LinkedList实现类
3.5.1 常用方法
  • offer()//添加元素在尾端
  • offerFirst/offerLast() //添加元素在首尾
  • poll() //删除链表头元素并且输出
  • pollFirst/pollLast() //删除链表头尾元素并且输出
  • remove()不能用在空链表的删除,而poll可以。
3.5.2遍历方式

除了前面三种方法,还有一种变形:

//这种方式比传统迭代器遍历更好的地方在于 对象随着for循环的结束而消亡,比传统迭代器更节省空间。
for(Iterator<String> it =List.Iterator();
it.hasnext();){
	System.out.println(it.next());
}
3.5.3底层实现

底层实现是用双向链表实现的。

在这里插入图片描述

3.5.4 面试题

1.Iterable,iterator(),Iterator之间的区别和联系
在这里插入图片描述
答:iterator()方法是Iterable接口中的一个抽象方法;

​ 而iterator()抽象方法在实现类中实现时,返回一个Iterator接口类型。

4.set接口

4.1特点

​ 相对于list接口,set接口中元素值唯一,并且无序,无序并不是随机。

4.2 遍历方式

由于set接口无序,所以不能有索引方法,因而不能使用普通for循环

  • 迭代器
  • 增强for循环
4.3 hashset实现类
  • 若使用add方法添加两个基本数据类型的相同元素,只会成功放入第一个

    底层原理
    在这里插入图片描述

  • 当填入自定义类型时,不满足唯一,无序,这是由于没有重写hashcode和equals方法

HashSet <Person> ps=new HashSet<>();
ps.add(new Person(16,"xiaoming"));
ps.add(new Person(16,"xiaohong"));
ps.add(new Person(16,"xiaoming"));
ps.add(new Person(17,"xiaohei"));
System.out.println(ps);

结果(没写前):[Person{age=16, name=‘xiaohong’}, Person{age=17, name=‘xiaohei’}, Person{age=16, name=‘xiaoming’}, Person{age=16, name=‘xiaoming’}]

结果(重写后):[Person{age=16, name=‘xiaoming’}, Person{age=16, name=‘xiaohong’}, Person{age=17, name=‘xiaohei’}]

4.4linkedhashset 实现类

实现唯一且有序,原理是:在所有元素上用链表串起来,每个元素位置固定。

4.5比较器
4.5.1内部比较器

1.String类实现了一个抽象方法compareTo()方法,该方法返回int类型值,实际原理:先比较两者长度,再用ASCII码值相减。

2.自定义类相比较时,需要该自定义类继承comparable接口,其中泛型为该类的类型,然后重写compareTo()方法,其中需要对比那个属性,参数就放入那个属性。

public class Student implements Comparable<Student> {
public int compareTo(Student o) {//重写编写排序规则
        if (this.age > o.getAge()) {
            return 1;
4.5.2外部比较器

自定义类外比较类继承comparator接口,重写compare()方法,该方法有连个参数,对应两个对象;

  • 外部比较器相对更优秀,多态,扩展性好;
public class Student implements Comparator<Student> {
public int compare(Student o1, Student o2) {
        return o1.getAge() > o2.getAge() ? 1 : (o1.getAge() == o2.getAge() ? 0 : -1);
    }

4.6 TreeSet 接口
4.6.1特点

元素唯一且有序(升序)

4.6.2底层

基本类型—>内部比较器

自定义类型—>外部比较器

4.6.3 遍历方式

中序遍历

4.6.4 TreeSet

TreeSet 放入元素

当利用外部比较器时,必须自己制定:

Comparator<Student> com =new BiJiao();
TreeSet<Student> ts=new TreeSet<>(com);
4.6 TreeSet 接口
4.6.1特点

元素唯一且有序(升序)

4.6.2底层

基本类型—>内部比较器

自定义类型—>外部比较器

4.6.3 遍历方式

中序遍历

4.6.4 TreeSet

TreeSet 放入元素

当利用外部比较器时,必须自己制定:

Comparator<Student> com =new BiJiao();
TreeSet<Student> ts=new TreeSet<>(com);

向TreeSet中放入数据时,自定义类必须实现比较器。

<think>嗯,用户问的是Java中的Collection单列集合。首先,我需要回忆一下Java集合框架的基础知识。记得Java集合框架主要分为Collection和Map两大类,而Collection是单列集合,Map是双列的键值对。用户可能想了解Collection的体系结构、常用接口和实现类,以及它们的使用场景。 首先,应该从Collection的顶层接口开始介绍,说明它继承了Iterable接口,所以所有Collection的实现类都支持迭代。然后,Collection分为List、Set和Queue三个子接口,每个子接口有不同的特性。 List部分,需要提到有序、可重复的特点,常用的实现类有ArrayList、LinkedList和Vector。要比较它们的底层实现,比如ArrayList基于数组,查询快,增删慢;LinkedList基于链表,增删快,查询慢;Vector是线程安全的,但性能较差,现在可能不常用了。 接下来是Set,强调元素唯一且无序。常用的实现类有HashSet、LinkedHashSet和TreeSet。HashSet基于哈希表,LinkedHashSet维护插入顺序,TreeSet可以排序。可能需要提到TreeSet是基于红黑树实现的,自然排序或者通过Comparator定制排序。 然后是Queue,队列,常用的有LinkedList、PriorityQueue,还有线程安全的ArrayBlockingQueue等。需要说明队列的先进先出特性,以及优先级队列的特殊之处。 用户可能还想知道如何选择不同的集合类,比如需要快速访问用ArrayList,需要频繁插入删除用LinkedList,需要唯一性用HashSet,需要排序用TreeSet,需要线程安全的话考虑使用Collections工具类或者并发包下的类。 另外,可能需要提到集合的遍历方式,比如for-each循环、迭代器、Java8的Stream API等。还要注意集合的线程安全问题,比如ArrayList不是线程安全的,多线程环境下应该用CopyOnWriteArrayList或者同步包装类。 用户可能的疑问还包括如何正确实现对象的equals和hashCode方法,特别是当使用HashSet或HashMap的时候。这也是确保集合正确行为的关键点。 最后,总结一下Collection的层次结构,以及各个实现类的适用场景,帮助用户在实际开发中做出合适的选择。可能还需要提醒一些常见错误,比如在遍历集合时修改导致的ConcurrentModificationException异常,以及如何避免。</think>Java中的Collection单列集合是存储一组对象的容器,属于Java集合框架的核心部分。以下从体系结构、常用接口和实现类进行分步解析: 1. **Collection顶层接口** - 继承自`Iterable`接口,支持迭代遍历 - 核心方法: ```java boolean add(E e); // 添加元素 boolean remove(Object o); // 删除元素 int size(); // 元素数量 boolean contains(Object o); // 判断包含 Iterator<E> iterator(); // 获取迭代器 ``` 2. **三大子接口** $$\begin{array}{|c|c|c|} \hline \text{接口} & \text{特点} & \text{常用实现类} \\ \hline \text{List} & \text{有序、可重复} & ArrayList, LinkedList, Vector \\ \text{Set} & \text{无序、唯一} & HashSet, LinkedHashSet, TreeSet \\ \text{Queue} & \text{队列结构} & LinkedList, PriorityQueue \\ \hline \end{array}$$ 3. **List接口详解** - **ArrayList**(数组实现): - 查询效率高($O(1)$随机访问) - 增删效率较低(需要移动元素) - **LinkedList**(双向链表实现): - 增删效率高($O(1)$头尾操作) - 查询效率低($O(n)$顺序访问) - **Vector**(线程安全数组): - 已被`Collections.synchronizedList()`取代 4. **Set接口详解** - **HashSet**(哈希表实现): - 基于`HashMap`实现 - 要求元素正确实现`hashCode()`和`equals()` - **LinkedHashSet**: - 维护插入顺序的HashSet - **TreeSet**(红黑树实现): - 支持自然排序或`Comparator`定制排序 - 元素必须实现`Comparable`接口 5. **Queue接口特性** - **PriorityQueue**: - 基于堆结构实现优先级队列 - 元素按自然顺序或Comparator排序 - **LinkedList**: - 可同时作为List和Deque使用 6. **使用示例** ```java // ArrayList示例 List<String> list = new ArrayList<>(); list.add("Java"); list.add("Python"); System.out.println(list.get(0)); // 输出:Java // HashSet示例 Set<Integer> set = new HashSet<>(); set.add(1); set.add(1); // 重复元素不会添加 System.out.println(set.size()); // 输出:1 // TreeSet排序示例 Set<String> sortedSet = new TreeSet<>(); sortedSet.add("Banana"); sortedSet.add("Apple"); System.out.println(sortedSet); // 输出:[Apple, Banana] ``` 7. **选择策略** - 需要快速随机访问 → ArrayList - 频繁插入删除 → LinkedList - 元素唯一性 → HashSet - 需要排序 → TreeSet - 线程安全需求 → 使用`Collections.synchronizedCollection()`包装或改用并发集合 8. **遍历方式** - 传统for循环(仅List) - 增强for循环 - Iterator迭代器 - Java8 Stream API: ```java list.stream().forEach(System.out::println); ``` 理解Collection体系需要掌握: 1. 各实现类的底层数据结构 2. 时间空间复杂度差异 3. 特定场景下的最佳实践 4. 线程安全问题的处理方式 建议通过实际编码练习加深理解,特别注意不同集合对元素相等性的判断机制,以及排序集合对元素可比性的要求。
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值