JAVA学习--集合Set使用

本文深入探讨了Java集合框架中的Collection、List、Set、Map等接口及其实现类,包括ArrayList、LinkedList、Vector、HashSet、LinkedHashSet、TreeSet等,详细解释了它们的特点、区别以及在实际开发中的应用。重点介绍了Set接口的无序性和不可重复性,通过HashSet、LinkedHashSet、TreeSet的使用案例,展示了如何利用哈希算法进行元素存储和排序。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


 * Collection接口 :
 *         |------List接口:
 *             |------ArrayList(主要的实现类)、
 *             |------LinkedList(对于频繁的插入、删除操作)、
 *             |------Vector(古老的实现类、线程安全的,但效率要低于ArrayList)
 *         |------Set接口:存储无序的,不可重复的元素.Set中常用的方法都是Collection下定义的。
                     |------HashSet(主要实现类)
 |------LinkedHashSet
 |------TreeSet
 

---------------------------------------------------------------------------------------------------------------
     * Set:存储的元素是无序的,不可重复的!
     1.无序性:无序性!= 随机性。真正的无序性,指的是元素在底层存储的位置是无序的。
     * 2.不可重复性:当向Set中添加进相同的元素的时候,后面的这个不能添加进去。
     *
     * 说明:要求添加进Set中的元素所在的类,一定要重写equals()和hashCode()方法。 进而保证Set中元素的不可重复性!
     *
     * Set中的元素时如何存储的呢?使用了哈希算法。
     * 当向Set中添加对象时,首先调用此对象所在类的hashCode()方法,计算此对象的哈希值,此哈希值
     * 决定了此对象在Set中的存储位置。若此位置之前没有对象存储,则这个对象直接存储到此位置。若此位置
     * 已有对象存储,再通过equals()比较这两个对象是否相同。如果相同,后一个对象就不能再添加进来。 万一返回false呢,都存储。(不建议如此)
     * >要求:hashCode()方法要与equals()方法一致。


    @Test
    public void testHashSet() {
        Set set = new HashSet();
        set.add(123);
        set.add(456);
        set.add(new String("AA"));
        set.add(new String("AA"));
        set.add("BB");
        set.add(null);//可以添加null
        Person p1 = new Person("GG", 23);
        Person p2 = new Person("GG", 23);
        System.out.println(p1.equals(p2));
        //可以观察添加hashCode方法和未添加后的差别
        System.out.println(p1.hashCode());
        System.out.println(p2.hashCode());
        set.add(p1);
        set.add(p2);
        System.out.println(set.size());
        System.out.println(set);
    }
--------------------------------------------------------------------------------------------------------------

     * LinkedHashSet:使用链表维护了一个添加进集合中的顺序。导致当我们遍历LinkedHashSet集合
     * 元素时,是按照添加进去的顺序遍历的!
     *
     * LinkedHashSet插入性能略低于 HashSet,但在迭代访问 Set 里的全部元素时有很好的性能。

    @Test
    public void testLinkedHashSet() {
        Set set = new LinkedHashSet();
        set.add(123);
        set.add(456);
        set.add(new String("AA"));
        set.add(new String("AA"));
        set.add("BB");
        set.add(null);

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


   
     * TreeSet: 1.向TreeSet中添加的元素必须是同一个类的。
     * 2.可以按照添加进集合中的元素的指定的顺序遍历。像String,包装类等默认按照从小到大的顺序遍历。
     * 3.当向TreeSet中添加自定义类的对象时,有两种排序方法:①自然排序②定制排序
     * 4.自然排序:要求自定义类实现java.lang.Comparable接口并重写其compareTo(Object obj)的抽象方法
     * 在此方法中,指明按照自定义类的哪个属性进行排序。
     *
     * 5.向TreeSet中添加元素时,首先按照compareTo()进行比较,一旦返回0,虽然仅是两个对象的此
     * 属性值相同,但是程序会认为这两个对象是相同的,进而后一个对象就不能添加进来。
     *
     * >compareTo()与hashCode()以及equals()三者保持一致!
    
    @Test
    public void testTreeSet1() {
        Set set = new TreeSet();
        // set.add(new String("AA"));
        // set.add(new String("AA"));
        // set.add("JJ");
        // set.add("GG");
        // set.add("MM");
        // 当Person类没有实现Comparable接口时,当向TreeSet中添加Person对象时,报ClassCastException
        set.add(new Person("CC", 23));
        set.add(new Person("MM", 21));
        set.add(new Person("GG", 25));
        set.add(new Person("JJ", 24));
        set.add(new Person("KK", 20));
        set.add(new Person("DD", 20));
        // set.add("AA");
        for (Object str : set) {
            System.out.println(str);
        }
    }
----------------------------------------------------------------------------------------------------------------
 
     * TreeSet的定制排序: 见下面的步骤 compare()与hashCode()以及equals()三者保持一致!

    @Test
    public void testTreeSet2() {
        // 1.创建一个实现了Comparator接口的类对象
        Comparator com = new Comparator() {
            // 向TreeSet中添加Customer类的对象,在此compare()方法中,指明是按照Customer
            // 的哪个属性排序的。
            @Override
            public int compare(Object o1, Object o2) {
                if (o1 instanceof Customer && o2 instanceof Customer) {
                    Customer c1 = (Customer) o1;
                    Customer c2 = (Customer) o2;
                    int i = c1.getId().compareTo(c2.getId());
                    if (i == 0) {
                        return c1.getName().compareTo(c2.getName());
                    }
                    return i;
                }
                return 0;
            }
        };
        // 2.将此对象作为形参传递给TreeSet的构造器中
        TreeSet set = new TreeSet(com);
        // 3.向TreeSet中添加Comparator接口中的compare方法中涉及的类的对象。
        set.add(new Customer("AA", 1003));
        set.add(new Customer("BB", 1002));
        set.add(new Customer("GG", 1004));
        set.add(new Customer("CC", 1001));
        set.add(new Customer("DD", 1001));

        for (Object str : set) {
            System.out.println(str);
        }
    }


    @Test
    public void testTreeSet3() {

        TreeSet set = new TreeSet(new Comparator() {
            public int compare(Object o1, Object o2) {
                if (o1 instanceof Customer && o2 instanceof Customer) {
                    Customer c1 = (Customer) o1;
                    Customer c2 = (Customer) o2;
                    int i = c1.getId().compareTo(c2.getId());
                    if (i == 0) {
                        return c1.getName().compareTo(c2.getName());
                    }
                    return i;
                }
                return 0;
            }
        });
        set.add(new Customer("AA", 1003));
        set.add(new Customer("BB", 1002));
        set.add(new Customer("GG", 1004));
        set.add(new Customer("CC", 1001));
        set.add(new Customer("DD", 1001));

        for (Object str : set) {
            System.out.println(str);
        }
    }


------------------------------------------------------------------以下为使用的类
class Customer {
    private String name;
    private Integer id;
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public Integer getId() {
        return id;
    }
    public void setId(Integer id) {
        this.id = id;
    }
    public Customer(String name, Integer id) {
        super();
        this.name = name;
        this.id = id;
    }
    public Customer() {
        super();
    }
    @Override
    public String toString() {
        return "Customer [name=" + name + ", id=" + id + "]";
    }
    @Override
    public int hashCode() {
        final int prime = 31;
        int result = 1;
        result = prime * result + ((id == null) ? 0 : id.hashCode());
        result = prime * result + ((name == null) ? 0 : name.hashCode());
        return result;
    }
    @Override
    public boolean equals(Object obj) {
        if (this == obj)
            return true;
        if (obj == null)
            return false;
        if (getClass() != obj.getClass())
            return false;
        Customer other = (Customer) obj;
        if (id == null) {
            if (other.id != null)
                return false;
        } else if (!id.equals(other.id))
            return false;
        if (name == null) {
            if (other.name != null)
                return false;
        } else if (!name.equals(other.name))
            return false;
        return true;
    }
   
}

class Person implements Comparable{
    private String name;
    private Integer 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;
    }
    public Person() {
        super();
    }
    public Person(String name, Integer age) {
        super();
        this.name = name;
        this.age = age;
    }
    @Override
    public String toString() {
        return "Person [name=" + name + ", age=" + age + "]";
    }
    //static int init = 1000;
    @Override
    public int hashCode() {//return age.hashCode() + name.hashCode();没下述的健壮性好。
        final int prime = 31;
        int result = 1;
        result = prime * result + ((age == null) ? 0 : age.hashCode());
        result = prime * result + ((name == null) ? 0 : name.hashCode());
        return result;
        //return init++;//不能这样用
    }
    @Override
    public boolean equals(Object obj) {
        if (this == obj)
            return true;
        if (obj == null)
            return false;
        if (getClass() != obj.getClass())
            return false;
        Person other = (Person) obj;
        if (age == null) {
            if (other.age != null)
                return false;
        } else if (!age.equals(other.age))
            return false;
        if (name == null) {
            if (other.name != null)
                return false;
        } else if (!name.equals(other.name))
            return false;
        return true;
    }
    //当向TreeSet中添加Person类的对象时,依据此方法,确定按照哪个属性排列。
    @Override
    public int compareTo(Object o) {
        if(o instanceof Person){
            Person p = (Person)o;
            //return this.name.compareTo(p.name);
            //return -this.age.compareTo(p.age);
            int i = this.age.compareTo(p.age);
            if(i == 0){
                return this.name.compareTo(p.name);
            }else{
                return i;
            }
        }
        return 0;
    }
   
}



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值