Java集合可用于存储数量不等的对象,并可以实现常用的数据结构,如栈,队列等。除此之外,Java集合还可用于保存具有映射关系的关联数组。

Java集合大致分为:List、Set、Queue、Map

List:代表有序、重复的集合

Set:代表无序、不可重复的集合

Queue:代表队列集合

Map:代表具有映射关系的集合(k-v)

为了保存数量不确定的数据,以及保存具有映射关系的数据(也被称为关联数组),Java提供了集合类,主要负责保存、盛装其他数据,因此也被称为容器类,所有的集合类都位于 java.util 包下。

集合类和数组不一样,数组是既可以存储基本数据类型,又可以存储引用数据类型,基本数据类型存储的是值,引用数据类型存储的是地址值,长度是固定的,不能自动增长;而集合只能存储引用数据类型(对象)集合中也可以存储基本数据类型,但是在存储的时候会自动装箱变成对象,长度是可变的,可以根据元素的增加而增长。

Java集合类主要由两个接口派生而出:Collection 和 Map,这两个接口是集合框架的根接口,其中又包括了一些子接口或实现类:

Java集合样式 java集合数据类型_数据结构

概述

  • List , Set, Map都是接口,前两个继承至Collection接口,Map为独立接口
  • Set下有HashSet,LinkedHashSet,TreeSet
  • List下有ArrayList,Vector,LinkedList
  • Map下有Hashtable,LinkedHashMap,HashMap,TreeMap
  • Collection接口下还有个Queue接口,有PriorityQueue类

Collection

注意:

  • Queue接口与List、Set同一级别,都是继承了Collection接口。 LinkedList既可以实现Queue接口,也可以实现List接口。只不过, LinkedList实现了Queue接口。Queue接口窄化了对LinkedList的方法的访问权限(即在方法中的参数类型如果是Queue时,就完全只能访问Queue接口所定义的方法 了,而不能直接访问 LinkedList的非Queue的方法),以使得只有恰当的方法才可以使用。
  • SortedSet是个接口,它里面的(只有TreeSet这一个实现可用)中的元素一定是有序的。

总结:

List 有序,可重复

  • ArrayList 优点: 底层数据结构是数组,查询快,增删慢。 缺点: 线程不安全,效率高
  • Vector 优点: 底层数据结构是数组,查询快,增删慢。 缺点: 线程安全,效率低
  • LinkedList 优点:底层数据结构是链表,查询慢,增删快。 缺点: 线程不安全,效率高

Set 无序,唯一

  • HashSet 底层数据结构是哈希表。(无序,唯一) 如何来保证元素唯一性? 1.依赖两个方法:hashCode()和equals()
  • LinkedHashSet 底层数据结构是链表和哈希表。(FIFO插入有序,唯一) 1.由链表保证元素有序 2.由哈希表保证元素唯一
  • TreeSet 底层数据结构是红黑树。(唯一,有序)
  1. 如何保证元素排序的呢? 自然排序 比较器排序
  2. 如何保证元素唯一性的呢? 根据比较的返回值是否是0来决定

 

TreeSet、LinkedHashSet、HashSet的区别

  • TreeSet的主要功能用于排序
  • LinkedHashSet的主要功能用于保证FIFO即有序的集合(先进先出)
  • HashSet只是通用的存储数据的集合

相同点:

  • Duplicates elements(重复元素): 因为三者都实现Set interface,所以三者都不包含duplicate elements
  • Thread safety(线程安全性): 三者都不是线程安全的,如果要使用线程安全可以Collections.synchronizedSet()

不同点:

  • Performance and Speed(性能和速度): HashSet插入数据最快,其次LinkHashSet,最慢的是TreeSet因为内部实现排序
  • Ordering(排序): HashSet不保证有序,LinkHashSet保证FIFO即按插入顺序排序,TreeSet安装内部实现排序,也可以自定义排序规则
  • null:HashSet和LinkHashSet允许存在null数据,但是TreeSet中插入null数据时会报NullPointerException

代码比较:

public static void main(String[] args) {
    HashSet<String> hashSet = new HashSet<>();
    LinkedHashSet<String> linkedHashSet = new LinkedHashSet<>();
    TreeSet<String> treeSet = new TreeSet<>();

    for (String data: Arrays.asList("1", "3", "5", "4", "2")) {
        hashSet.add(data);
        linkedHashSet.add(data);
        treeSet.add(data);
    }

    //不保证有序
    System.out.println("Ordering in HashSet :" + hashSet + "(无顺序)");

    //FIFO保证安装插入顺序排序
    System.out.println("Order of element in LinkedHashSet :" + linkedHashSet + "(FIFO插入有序)");

    //内部实现排序
    System.out.println("Order of objects in TreeSet :" + treeSet + "(排序)");
}

输出结果:

Ordering in HashSet :[1, 2, 3, 4, 5](无顺序)
Order of element in LinkedHashSet :[1, 3, 5, 4, 2](FIFO插入有序)
Order of objects in TreeSet :[1, 2, 3, 4, 5](排序)

性能

Java集合样式 java集合数据类型_java_02

TreeSet的两种排序方式比较

  1. 排序的引入(以基本数据类型的排序为例)
    由于TreeSet可以实现对元素按照某种规则进行排序,例如下面的例子
public static void main(String[] args) {
    // 创建集合对象
    // 自然顺序进行排序
    TreeSet<Integer> treeSet = new TreeSet<>();

    // 创建元素并添加
    // 20,18,23,22,17,24,19,18,24
    treeSet.add(20);
    treeSet.add(18);
    treeSet.add(23);
    treeSet.add(22);
    treeSet.add(17);
    treeSet.add(24);
    treeSet.add(19);
    treeSet.add(18);
    treeSet.add(24);

    // 遍历
    for (Integer i : treeSet) {
        System.out.println(i);
    }
}

输出结果:

17
18
19
20
22
23
24
  1. 如果是引用数据类型,比如自定义对象
    测试类:
public class TestSet {
    public static void main(String[] args) {
        TreeSet<Student> ts = new TreeSet<Student>();
        //创建元素对象
        Student s1=new Student("zhangsan",20);
        Student s2=new Student("lis",22);
        Student s3=new Student("wangwu",24);
        Student s4=new Student("chenliu",26);
        Student s5=new Student("zhangsan",22);
        Student s6=new Student("qianqi",24);

        //将元素对象添加到集合对象中
        ts.add(s1);
        ts.add(s2);
        ts.add(s3);
        ts.add(s4);
        ts.add(s5);
        ts.add(s6);

        //遍历
        for(Student s:ts){
            System.out.println(s.getName()+"-----------"+s.getAge());
        }
    }
}
class Student{
    private String name;
    private Integer age;

    public Student() {
    }

    public Student(String name, Integer age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }
}

运行结果:

Exception in thread "main" java.lang.ClassCastException: com.tx.lesson.Student cannot be cast to java.lang.Comparable
    at java.util.TreeMap.compare(TreeMap.java:1290)
    at java.util.TreeMap.put(TreeMap.java:538)
    at java.util.TreeSet.add(TreeSet.java:255)
    at com.tx.lesson.TestSet.main(TestSet.java:20)

原因分析: 由于不知道该安照那一中排序方式排序,所以会报错。 解决方法:

自然排序

Student类中实现 Comparable接口

Comparable是排序接口。若一个类实现了Comparable接口,就意味着该类支持 排序。实现了Comparable接口的类的对象的列表或数组可以通过Collections.sort或 Arrays.sort进行自动排序。 此外,实现此接口的对象可以用作有序映射中的键或有序集合中的集合,无需指 定比较器。 此接口只有一个方法compare,比较此对象与指定对象的顺序,如果该对象小 于、等于或大于指定对象,则分别返回负整数、零或正整数。

重写Comparable接口中的Compareto方法(compareTo(T o) 比较此对象与指定对象的顺序

class Student implements Comparable<Student>{
    private String name;
    private Integer age;

    public Student() {
    }

    public Student(String name, Integer age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }

    @Override
    public int compareTo(Student s) {
        //return -1; //-1表示放在红黑树的左边,即逆序输出
        //return 1;  //1表示放在红黑树的右边,即顺序输出
        //return o;  //表示元素相同,仅存放第一个元素
        //主要条件 姓名的长度,如果姓名长度小的就放在左子树,否则放在右子树
        int num=this.name.length()-s.name.length();
        //姓名的长度相同,不代表内容相同,如果按字典顺序此 String 对象位于参数字符串之前,则比较结果为一个负整数。
        //如果按字典顺序此 String 对象位于参数字符串之后,则比较结果为一个正整数。
        //如果这两个字符串相等,则结果为 0
        int num1=num==0?this.name.compareTo(s.name):num;
        //姓名的长度和内容相同,不代表年龄相同,所以还要判断年龄
        int num2=num1==0?this.age-s.age:num1;
        return num2;
    }
}

输出结果:

lis-----------22
qianqi-----------24
wangwu-----------24
chenliu-----------26
zhangsan-----------20
zhangsan-----------22

比较器排序

单独创建一个比较类,这里以MyComparator为例,并且要让其继承Comparator接口;

Comparator是比较接口,我们如果需要控制某个类的次序,而该类本身不支持排 序(即没有实现Comparable接口),那么我们就可以建立一个“该类的比较器”来进行排 序,这个“比较器”只需要实现Comparator接口即可。也就是说,我们可以通过实现 Comparator来新建一个比较器,然后通过这个比较器对类进行排序。

注意: 若一个类要实现Comparator接口:它一定要实现compare(T o1, T o2) 函数,但 可以不实现 equals(Object obj) 函数。 int compare(T o1, T o2) 是“比较o1和o2的大小”。返回“负数”,意味着“o1比o2 小”;返回“零”,意味着“o1等于o2”;返回“正数”,意味着“o1大于o2”。

重写Comparator接口中的Compare方法(compare(T o1,T o2) 比较用来排序的两个参数。

在主类中使用下面的构造方法(TreeSet(Comparator<? superE> comparator)构造一个新的空TreeSet,根据指定比较器进行排序

public class TestSet {
    public static void main(String[] args) {
        TreeSet<Student> ts = new TreeSet<Student>(new MyComparator());
        //创建元素对象
        Student s1=new Student("zhangsan",20);
        Student s2=new Student("lis",22);
        Student s3=new Student("wangwu",24);
        Student s4=new Student("chenliu",26);
        Student s5=new Student("zhangsan",22);
        Student s6=new Student("qianqi",24);

        //将元素对象添加到集合对象中
        ts.add(s1);
        ts.add(s2);
        ts.add(s3);
        ts.add(s4);
        ts.add(s5);
        ts.add(s6);

        //遍历
        for(Student s:ts){
            System.out.println(s.getName()+"-----------"+s.getAge());
        }
    }
}
class Student{
    private String name;
    private Integer age;

    public Student() {
    }

    public Student(String name, Integer age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }
}
class MyComparator implements Comparator<Student>{
    @Override
    public int compare(Student s1, Student s2) {
        // 姓名长度
        int num = s1.getName().length() - s2.getName().length();
        // 姓名内容
        int num2 = num == 0 ? s1.getName().compareTo(s2.getName()) : num;
        // 年龄
        int num3 = num2 == 0 ? s1.getAge() - s2.getAge() : num2;
        return num3;
    }
}

Comparable和Comparator区别比较

Comparable是排序接口,若一个类实现了Comparable接口,就意味着“该类支持 排序”。Comparator是比较器,我们若需要控制某个类的次序,可以建立一个“该类的 比较器”来进行排序。Comparable相当于“内部比较器”,而Comparator相当于“外部比较器”。

用Comparable简单, 只要实现Comparable 接口的对象直接就成为一个可以比较 的对象,但是需要修改源代码。 用Comparator 的好处是不需要修改源代码, 而是另 外实现一个比较器, 当某个自定义的对象需要作比较的时候,把比较器和对象一起传 递过去就可以比大小了, 并且在Comparator 里面用户可以自己实现复杂的可以通用 的逻辑,使其可以匹配一些比较简单的对象,那样就可以节省很多重复劳动了。

Map

Map接口有三个比较重要的实现类,分别是HashMap、TreeMap和HashTable。

  • TreeMap是有序的,HashMap和HashTable是无序的。
  • Hashtable的方法是同步的,HashMap的方法不是同步的。这是两者最主要的区别。

这就意味着:

  • Hashtable是线程安全的,HashMap不是线程安全的。
  • HashMap效率较高,Hashtable效率较低。 如果对同步性或与遗留代码的兼容性没有任何要求,建议使用HashMap。 查看Hashtable的源代码就可以发现,除构造函数外,Hashtable的所有 public 方法声明中都有 synchronized关键字,而HashMap的源码中则没有。
  • Hashtable不允许null值,HashMap允许null值(key和value都允许)
  • 父类不同:Hashtable的父类是Dictionary,HashMap的父类是AbstractMap