单列集合Set的实现类HashSet

Set接口 [Collection】的子类
HashSet

特点【无序,不可重复,不能排序】
默认比较地址值【地址相同的值相同】,重写后可比较内容【内容相同的值相同】

1.比较地址值【默认】

public class Demo1 {
    public static void main(String[] args) {
        HashSet<Student> hs = new HashSet<>();
        Student s1 = new Student("张三", 18);
        Student s2 = new Student("李四", 18);
        Student s3 = new Student("张三", 18);
        System.out.println("s1:"+ s1.hashCode());
        System.out.println("s2:"+ s2.hashCode());
        System.out.println("s3:"+ s3.hashCode());
        System.out.println("s1:"+ s1.hashCode());
        hs.add(s1);
        hs.add(s2);
        hs.add(s3);
        for(Student student : hs){
            System.out.println(student);
        }
    }
}

打印结果:
--------------------------------------------------------------
s1:189568618
s2:960604060
s3:1349393271
s1:189568618
Student{name='张三', age=18}
Student{name='张三', age=18}
Student{name='李四', age=18}

2.而默认比较地址值不能达到我们实际的要求,则需要重写HashaCode方法使其比较属性值

【API中有一些类已经重写了HashaCode,给出了比较规则,如:String:数值大小、字符串长度 ]

public class Demo1 {
    public static void main(String[] args) {
        HashSet<String> hs = new HashSet<>();
        hs.add("hello");
        hs.add("world");
        hs.add("java");
        hs.add("java");
        hs.add("java");
        for (String h : hs) {
            System.out.println(h);
        }
    }
}
打印结果:
--------------------------------------------------------------
world
java
hello

【未重写的类,如:我们自定义的类,则需要去类中重写HashC ode方法】

public class Demo1 {
    public static void main(String[] args) {
        HashSet<Student1> hs = new HashSet<>();
        Student1 s1 = new Student1("张三", 18);
        Student1 s2 = new Student1("李四", 18);
        Student1 s3 = new Student1("张三", 18);
        hs.add(s1);
        hs.add(s2);
        hs.add(s3);
        for(Student1 student : hs){
            System.out.println(student);
        }
    }
}

class Student {
    private String name;
    private int  age;

    public Student(String name, int age) {
        this.name = name;
        this.age = age;
    }
    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;

        Student student = (Student) o;

        if (age != student.age) return false;
        return Objects.equals(name, student.name);
    }
    @Override
    public int hashCode() {
        int result = name != null ? name.hashCode() : 0;
        result = 31 * result + age;
        return result;
    }
}
打印结果:
------------------------------------------------------------
Student{name='张三', age=18}
Student{name='李四', age=18}

3.底层原理【链表,数组,[链表满8为]红黑树】

计算哈希值 : 根据元素的哈希数和数组的长度取余得到存入的位置 如:21501220%16
在这里插入图片描述

注:两种Set实现类的选择:【如果你需要得到一个有序的结果时就应该使用TreeMap(因为HashMap中元素的排列顺序是不固定的)。除此之外,由于HashMap有更好的性能,所以大多不需要排序的时候我们会使用HashMap】

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

陪雨岁岁年年

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值