Java集合-Set接口

Collection接口:单列集合,用来存储一个一个的对象
* (不常用)子接口Set:存储无序的、不可重复的数据
* HashSet:作为Set接口的主要实现类;线程不安全的;可以存储null值
* LinkedHashSet:作为HashSet的子类,遍历期内部数据时,可以按照添加的顺序遍历
* TreeSet:可以按照添加对象的指定属性,进行排序

Set接口
* 无序性:不等于随机性,无序性相当于List而言,存储的数据在底层数组中并非按照数组索引的顺序添加,而是根据数据的哈希值决定的
* 不可重复性:保证添加的元素按照equals()判断时,不能返回true。即:相同元素只能添加一个;(类需要重写equals和hashCode)
*
* 1.Set接口中没有额外定义的新的方法,使用的都是Collection中声明的方法。
*
* 2.添加元素的过程
* 我们向HashSet中添加元素a,首先调用元素a所在类的hashCode()方法,计算a的哈希值,
* 此哈希值接着通过某种算法计算出在HashSet底层数组中存放的位置(即为:索引位置),判断
* 此数组此位置上是否有元素。
* 如果此位置上没有其他对象元素,则元素a添加成功;//情况一
* 如果此位置上有其他元素b(或以链表形式存在的多个元素),则比较元素a与元素b的hash值:
* 如果hash值不同,则元素a添加成功。--->情况二
* 如果hash值相同,进而调用元素a所在类的equals()方法:
* equals()返回true,元素a添加失败。---->情况三
* equals()返回false,元素a添加成功。---->情况四
* 对于添加成功的情况二、四而言,元素a与已经存在指定索引位置上的数据以链表的形式存储。
* jdk 7:元素a放到数组中,指向原来的元素
* jdk 8:原来的元素放到数组中,指向元素a
* 总结:七上八下
*
* HashSet底层:数组+链表的结构


import org.junit.Test;

import java.util.*;

/**
 *
 *  Collection接口:单列集合,用来存储一个一个的对象
 *      (不常用)子接口Set:存储无序的、不可重复的数据
 *              HashSet:作为Set接口的主要实现类;线程不安全的;可以存储null值
 *                  LinkedHashSet:作为HashSet的子类,遍历期内部数据时,可以按照添加的顺序遍历
 *              TreeSet:可以按照添加对象的指定属性,进行排序
 * @author orz
 */
public class SetTest {

    /**
     *Set接口
     * 无序性:不等于随机性,无序性相当于List而言,存储的数据在底层数组中并非按照数组索引的顺序添加,而是根据数据的哈希值决定的
     * 不可重复性:保证添加的元素按照equals()判断时,不能返回true。即:相同元素只能添加一个;(类需要重写equals和hashCode)
     *
     * 1.Set接口中没有额外定义的新的方法,使用的都是Collection中声明的方法。
     *
     * 2.添加元素的过程
     *  我们向HashSet中添加元素a,首先调用元素a所在类的hashCode()方法,计算a的哈希值,
     *  此哈希值接着通过某种算法计算出在HashSet底层数组中存放的位置(即为:索引位置),判断
     *  此数组此位置上是否有元素。
     *      如果此位置上没有其他对象元素,则元素a添加成功;//情况一
     *      如果此位置上有其他元素b(或以链表形式存在的多个元素),则比较元素a与元素b的hash值:
     *          如果hash值不同,则元素a添加成功。--->情况二
     *          如果hash值相同,进而调用元素a所在类的equals()方法:
     *              equals()返回true,元素a添加失败。---->情况三
     *              equals()返回false,元素a添加成功。---->情况四
     *  对于添加成功的情况二、四而言,元素a与已经存在指定索引位置上的数据以链表的形式存储。
     *  jdk 7:元素a放到数组中,指向原来的元素
     *  jdk 8:原来的元素放到数组中,指向元素a
     *  总结:七上八下
     *
     *  HashSet底层:数组+链表的结构
     *
     */

    @Test
    public void test1()
    {

        HashSet hashSet = new HashSet();
        hashSet.add("AA");
        hashSet.add("BB");
        hashSet.add("CC");
        hashSet.add("CC");
        hashSet.add("CC");
        Iterator iterator = hashSet.iterator();
        while (iterator.hasNext())
        {
            System.out.println(iterator.next());
        }

    }


    /**
     * LinkedHashSet:可以按照添加的顺序遍历
     * LinkedHashSet:作为HashSet的子类,再添加数据的同时,每个数据还维护了两个引用,记录此数据前一个数据和后一个数据
     * 优点:对于频繁的遍历操作,效率高于HashSet
     *
     */
    @Test
    public void test2()
    {

        HashSet hashSet2 = new HashSet();
        hashSet2.add(123);
        hashSet2.add(234);
        hashSet2.add("AA");
        hashSet2.add("BB");
        hashSet2.add("CC");

        Iterator iterator2 = hashSet2.iterator();
        while (iterator2.hasNext())
        {
            System.out.println(iterator2.next());
        }
        System.out.println();

        //可以按照添加的顺序遍历
        HashSet hashSet = new LinkedHashSet();
        hashSet.add(123);
        hashSet.add(234);
        hashSet.add("AA");
        hashSet.add("BB");
        hashSet.add("CC");

        Iterator iterator = hashSet.iterator();
        while (iterator.hasNext())
        {
            System.out.println(iterator.next());
        }

    }


    /**TreeSet和TreeMap采用红黑树的存储结构
     * TreeSet:
     * 1.向TreeSet中添加的数据,要求是相同的对象
     * 2.两种排序方式:
     *  自然排序(实现Comparable接口):比较两个对象是否相同的标准为:compareTo()返回0,不再是equals();
     *
     *  定制排序(Comparator):比较两个对象是否相同的标准为:compare()返回0,不再是equals();
     */
    @Test
    public void test3()
    {
        TreeSet treeSet = new TreeSet();
        //错误,不能添加不同类的对象
//        treeSet.add(123);
//        treeSet.add(456);
//        treeSet.add("AA");

        treeSet.add(new Person(21,"Tom"));
        treeSet.add(new Person(23,"Jack"));
        treeSet.add(new Person(25,"Jack"));
        treeSet.add(new Person(43,"Cat"));
        treeSet.add(new Person(12,"Dog"));

        Iterator iterator = treeSet.iterator();
        while (iterator.hasNext())
        {
            System.out.println(iterator.next());
        }


    }

    @Test
    public void test4()
    {
        Comparator com=new Comparator() {
            @Override
            public int compare(Object o1, Object o2) {
                if(o1 instanceof Person && o2 instanceof Person)
                {
                    Person person=(Person)o1;
                    Person person1=(Person)o2;
                    return Integer.compare(person.getAge(),person1.getAge());
                }
                else
                {
                    throw new RuntimeException("输入的数据类型不匹配");
                }

            }
        };
        TreeSet treeSet = new TreeSet(com);
        treeSet.add(new Person(21,"Tom"));
        treeSet.add(new Person(23,"Jack"));
        treeSet.add(new Person(25,"Jack"));
        treeSet.add(new Person(43,"Cat"));
        treeSet.add(new Person(12,"Dog"));



        Iterator iterator = treeSet.iterator();
        while (iterator.hasNext())
        {
            System.out.println(iterator.next());
        }

    }

}

 

import java.util.Objects;

public class Person implements Comparable{

    private int age;
    private String name;

    public Person() {
    }

    public Person(int age, String name) {
        this.age = age;
        this.name = name;
    }

    public int getAge() {
        return age;
    }

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

    public String getName() {
        return name;
    }

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

    @Override
    public String toString() {
        return "Person{" +
                "age=" + age +
                ", name='" + name + '\'' +
                '}';
    }

    @Override
    public boolean equals(Object o) {
        if (this == o) {
            return true;
        }
        if (o == null || getClass() != o.getClass()) {
            return false;
        }
        Person person = (Person) o;
        return age == person.age &&
                Objects.equals(name, person.name);
    }

    @Override
    public int hashCode() {
        return Objects.hash(age, name);
    }

    //按照姓名从小到大排序,年龄从小到大

   @Override
   public int compareTo(Object o)
   {
       if(o instanceof Person)
       {
           Person person=(Person)o;
          // return this.name.compareTo(person.name);
           int compare=this.name.compareTo(person.name);
           if(compare!=0)
           {
               return compare;
           }
           else
           {
               return Integer.compare(this.age,person.age);
           }

       }
       else
       {
           throw new RuntimeException("输入的类型不匹配");
       }
   }
}

 

posted @ 2020-08-10 22:32  orz江小鱼  阅读(148)  评论(0编辑  收藏  举报