Collection子接口:Set接口

1.Set 存储的数据特点:无序的、不可重复的元素
具体的:HashSet为例说明:
  1. 无序性:不等于随机性。存储的数据在底层数组中并非照数组索引的顺序添加而是根据数据的哈希值决定的
  2. 不可重复性:保证添加的元素照equals()判断时,不能返回true.即:相同的元素只能添加一个

2. 元素添加过程:(以HashSet为例)---HashSet底层:数组+链表的结构。(前提:jdk7)
我们向HashSet中添加元素a,首先调用元素a所在类的hashCode()方法,计算元素a的哈希值,此哈希值接着通过某种算法计算出在HashSet底层数组中的存放位置(即为:索引位置,判断数组此位置上是否已经元素:
    如果此位置上没其他元素,则元素a添加成功。 --->情况1
    如果此位置上其他元素b(或以链表形式存在的多个元素,则比较元素a与元素b的hash值:
        如果hash值不相同,则元素a添加成功。--->情况2
        如果hash值相同,进而需要调用元素a所在类的equals()方法:
               equals()返回true,元素a添加失败
               equals()返回false,则元素a添加成功。--->情况3
对于添加成功的情况2和情况3而言:元素a 与已经存在指定索引位置上数据以链表的方式存储。
jdk 7 :元素a放到数组中,指向原来的元素。
jdk 8 :原来的元素在数组中,指向元素a
总结:七上八下
 
3. 常用方法
Set接口中没额外定义新的方法,使用的都是Collection中声明过的方法。
 
4. 常用实现类:
|----Collection接口:单列集合,用来存储一个一个的对象
*          |----Set接口:存储无序的、不可重复的数据   -->高中讲的“集合”
*              |----HashSet:作为Set接口的主要实现类;线程不安全的;可以存储null值
*                  |----LinkedHashSet:作为HashSet的子类;遍历其内部数据时,可以按照添加的顺序遍历
*                 在添加数据的同时,每个数据还维护了两个引用,记录此数据前一个数据和后一个数据。                  
       对于频繁的遍历操作,LinkedHashSet效率高于HashSet.
*              |----TreeSet:可以照添加对象的指定属性,进行排序。
 
5. 存储对象所在类的要求:
  HashSet/LinkedHashSet:
要求:Set(主要指:HashSet、LinkedHashSet)中添加的数据,其所在的类一定要重写hashCode() 和 equals()
为什么一定要重写hashCode() 和 equals():不全面的理解为默认情况下,hashCode返回的就是对象的存储地址,这样即使相同的对象也会因为不同的内存地址而被加入到Set中,但这违背Set的定义(不可重复)。因此一定要重写hashCode。此外,哈希值相同不一定是相同对象,因为存在碰撞冲突,因此还需要通过equals判断是否对象属性一致。但是不同的哈希值一定对应不同的对象。
要求:重写的hashCode()和equals()尽可能保持一致性:相等的对象必须具有相等的散列码
*    重写两个方法的小技巧:对象中用作 equals() 方法比较的 Field,都应该用来计算 hashCode 值。
*
  TreeSet:
要求:如果试图把一个对象添加到 TreeSet 时,则该对象的类必须实现 Comparable 接口。
1.自然排序中,比较两个对象是否相同的标准为:compareTo()返回0.
2.定制排序中,比较两个对象是否相同的标准为:compare()返回0. 需要将实现Comparator接口的实例作为形参传递给TreeSet的构造器。
  向 TreeSet 中添加元素时,只有第一个元素无须比较compareTo()方法,后面添 加的所有元素都会调用compareTo()方法进行比较。
  因为只有相同类的两个实例才会比较大小,所以向 TreeSet 中添加的应该是同 一个类的对象。
  对于 TreeSet 集合而言,它判断两个对象是否相等的唯一标准是:两个对象通 过 compareTo(Object obj) 方法比较返回值。
  当需要把一个对象放入 TreeSet 中,重写该对象对应的 equals() 方法时,应保 证该方法与 compareTo(Object obj) 方法有一致的结果:如果两个对象通过 equals() 方法比较返回 true,则通过 compareTo(Object obj) 方法比较应返回 0。 否则,让人难以理解。

Java比较器的使用背景:
Java中的对象,正常情况下,只能进行比较:==  或  != 。不能使用 > 或 < 的。但是在开发场景中,我们需要对多个对象进行排序,言外之意,就需要比较对象的大小。
  如何实现?使用两个接口中的任何一个:Comparable 或 Comparator
2.自然排序:使用Comparable接口
 2.1 说明
  1.像String、包装类等实现了Comparable接口,重写了compareTo(obj)方法,给出了比较两个对象大小的方式。
  2.像String、包装类重写compareTo()方法以后,进行了从小到大的排列
 3. 重写compareTo(obj)的规则:
     如果当前对象this大于形参对象obj,则返回正整数,
     如果当前对象this小于形参对象obj,则返回负整数,
     如果当前对象this等于形参对象obj,则返回零。
 4. 对于自定义类来说,如果需要排序,我们可以让自定义类实现Comparable接口,重写compareTo(obj)方法。在compareTo(obj)方法中指明如何排序
2.2 自定义类代码举例:
public class Goods implements  Comparable{

    private String name;
    private double price;

    //指明商品比较大小的方式:照价格从低到高排序,再照产品名称从高到低排序
    @Override
    public int compareTo(Object o) {
//        System.out.println("**************");
        if(o instanceof Goods){
            Goods goods = (Goods)o;
            //方式一:
            if(this.price > goods.price){
                return 1;
            }else if(this.price < goods.price){
                return -1;
            }else{
//                return 0;
               return -this.name.compareTo(goods.name);
            }
            //方式二:
//           return Double.compare(this.price,goods.price);
        }
//        return 0;
        throw new RuntimeException("传入的数据类型不一致!");
    }
// getter、setter、toString()、构造器:省略
}
3.定制排序:使用Comparator接口
  3.1 说明
   1.背景:当元素的类型没实现java.lang.Comparable接口而又不方便修改代码,或者实现了java.lang.Comparable接口的排序规则不适合当前的操作,那么可以考虑使用 Comparator 的对象来排序
   2.重写compare(Object o1,Object o2)方法,比较o1和o2的大小:
    如果方法返回正整数,则表示o1大于o2;
    如果返回0,表示相等;
    返回负整数,表示o1小于o2。
  3.2 代码举例:
Comparator com = new Comparator() {
    //指明商品比较大小的方式:照产品名称从低到高排序,再照价格从高到低排序
    @Override
    public int compare(Object o1, Object o2) {
        if(o1 instanceof Goods && o2 instanceof Goods){
            Goods g1 = (Goods)o1;
            Goods g2 = (Goods)o2;
            if(g1.getName().equals(g2.getName())){
                return -Double.compare(g1.getPrice(),g2.getPrice());
            }else{
                return g1.getName().compareTo(g2.getName());
            }
        }
        throw new RuntimeException("输入的数据类型不一致");
    }
}

使用:

  Arrays.sort(goods,com);
  Collections.sort(coll,com);
  new TreeSet(com);
4. 两种排序方式对比
*    Comparable接口的方式一旦一定,保证Comparable接口实现类的对象在任何位置都可以比较大小。
*    Comparator接口属于临时性的比较。

6. TreeSet的使用
  6.1 使用说明:
    1.向TreeSet中添加的数据,要求是相同类的对象
    2.两种排序方式:自然排序(实现Comparable接口和定制排序(Comparator)

  6.2用的排序方式:
//方式一:自然排序
@Test
    public void test1(){
        TreeSet set = new TreeSet();

        //失败:不能添加不同类的对象
//        set.add(123);
//        set.add(456);
//        set.add("AA");
//        set.add(new User("Tom",12));

            //举例一:
//        set.add(34);
//        set.add(-34);
//        set.add(43);
//        set.add(11);
//        set.add(8);

        //举例二:
        set.add(new User("Tom",12));
        set.add(new User("Jerry",32));
        set.add(new User("Jim",2));
        set.add(new User("Mike",65));
        set.add(new User("Jack",33));
        set.add(new User("Jack",56));


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

    }

//方式二:定制排序
    @Test
    public void test2(){
        Comparator com = new Comparator() {
            //照年龄从小到大排列
            @Override
            public int compare(Object o1, Object o2) {
                if(o1 instanceof User && o2 instanceof User){
                    User u1 = (User)o1;
                    User u2 = (User)o2;
                    return Integer.compare(u1.getAge(),u2.getAge());
                }else{
                    throw new RuntimeException("输入的数据类型不匹配");
                }
            }
        };

        TreeSet set = new TreeSet(com);
        set.add(new User("Tom",12));
        set.add(new User("Jerry",32));
        set.add(new User("Jim",2));
        set.add(new User("Mike",65));
        set.add(new User("Mary",33));
        set.add(new User("Jack",33));
        set.add(new User("Jack",56));


        Iterator iterator = set.iterator();
        while(iterator.hasNext()){
            System.out.println(iterator.next());
        }
    }
package TreeSetTest;

public class User implements Comparable {
    private  String name;
    private  int age;

    public User() {
    }

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

    public String getName() {
        return name;
    }

    public int getAge() {
        return age;
    }

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

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

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


    @Override
    public int compareTo(Object o) {
        if(o instanceof  User){
            User u=(User) o;
            int compare=-this.name.compareTo(u.name); //String、包装类等实现了Comparable接口,重写了compareTo(obj)方法,进行了从小到大的排列
            if(compare!=0){
                return  compare;
            }else{
                return -Integer.compare(u.age,this.age);//对于基本数据类型,采用包装类的compare方法
                //return Integer.valueOf(u.age).compareTo(Integer.valueOf(this.age));
            }
        }
        else {
            throw  new RuntimeException("输入类型不匹配");
        }

    }

    @Override
    public boolean equals(Object obj) {
        System.out.println("equals");
        if(this==obj)
            return  true;
        if(obj==null || getClass()!=obj.getClass()) return false;

        User user=(User) obj;
//        if(this.name.equals(u.name) && this.age==((User) obj).age)
//            return true;
//        else
//            return false;
        if (age != user.age) return false;
        return name != null ? name.equals(user.name) : user.name == null;
    }

    @Override
    public int hashCode() {
        int result= name!=null? name.hashCode():0;
        result=31*result+age;
        return  result;
    }
}

参考

 
关于底层实现:

 

 

 

 

 

 
posted @ 2020-05-04 16:40  kkzhang  阅读(198)  评论(0编辑  收藏  举报