java详解 --- Set接口

本文深入探讨了Java集合框架中的Set接口及其子类,包括HashSet、LinkedHashSet和TreeSet的使用方法与特点。通过实例展示了如何利用这些集合去除重复元素、排序及实现自定义对象的比较。
摘要由CSDN通过智能技术生成

set接口是继承于collection接口,是collection的一个分支.

一.hashSet

1.hashSet是可以用来去除集合中的重复的元素
代码举例:

HashSet<String> hashSet = new HashSet<>();
    hashSet.add("a");
    hashSet.add("a");
    hashSet.add("b");
    hashSet.add("b");
    hashSet.add("c");
    hashSet.add("c");
    hashSet.add("d");
    hashSet.add("d");
    // 使用增强for循环 遍历
    for(String string : hashSet){
        System.out.println(string);
    }
    System.out.println(hashSet);

结果为:
这里写图片描述

2.LinkedHashSet是hashSet的子类,特点是去重 有序排序,怎么存的怎么取出来
代码举例:

LinkedHashSet<String> linkedHashSet = new LinkedHashSet<>();
    linkedHashSet.add("a");
    linkedHashSet.add("a");
    linkedHashSet.add("b");
    linkedHashSet.add("b");
    linkedHashSet.add("c");
    linkedHashSet.add("c");
    linkedHashSet.add("d");
    linkedHashSet.add("d");
    System.out.println(linkedHashSet);

输出结果:
这里写图片描述

3.问题举例:
①.需求:编写一个程序 获取10个1 - 20的随机数.要求不能重复

HashSet<Integer> hashSet = new HashSet<>();
// while写一个死循环
while(true){
    // 判断集合长度超过10,结束循环
    if(hashSet >= 10){
        break;
    }
    // 将随机数添加进hashset中
    int num = (int)(Math.random() * (20 + 1 - 1 ) + 1);
    hashSet.add(num);
}
System.out.println(hashSet);

输出结果:
这里写图片描述
②.输入一个字符串,去掉其中重复字符, 打印出不同的那些字符

// 接收字符串
System.out.println("请输入一个字符串:");
Scanner scanner = new Scanner(System.in);
String string = scanner.nextLine();
// 将字符串转化为字符数组
char[] array = string.toCharArray();
// 创建set集合
HashSet<Character> hashSet = new HashSet<>();
// 将字符数组放进set集合
for(int i = 0 ; i < array.length ; i++) {
    hashSet.add(array[i]);
}
System.out.println(hashSet);

输出结果:
这里写图片描述
③.利用set集合 去除ArrayList集合中的重复元素(操作原ArrayList集合)

ArrayList<String> arrayList = new ArrayList<>();
    arrayList.add("a");
    arrayList.add("a");
    arrayList.add("b");
    arrayList.add("b");
    arrayList.add("c");
    arrayList.add("c");
// 保持怎么存怎么取的原则 使用linkedHashSet
LinkedHashSet<String> linkedHashSet = new LinkedHashSet<>();
// 将arrayList中的元素添加到LinkedHashSet
linkedHashSet.addAll(arrayList);
// 清空arrayList集合
arrayList.clear();
// 将LinkedHashSet中的元素添加到arrayList
arrayList.addAll(linkedHashSet);
System.out.println(arrayList);

输出结果:
这里写图片描述
④.创建一个hashset 保存5个人 遍历
认为年龄和姓名一样就是同一个人 去除重复的
先创建一个人类:

public class Person implements Comparable<Person> {
    private String name;
    private int age;

    // 构造方法
    public Person() {

    }

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

    // set/get方法
    public String getName() {
        return name;
    }

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

    public int getAge() {
        return age;
    }

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

    // toString方法的重写
    @Override
    public String toString() {
        return "Person [name=" + name + ", age=" + age + "]";
    }

    // 生成哈希值
    // 重写hashcode方法
    // hashcode在存储对象时 按照哈希值去存储的
    // 注意 保存对象的时候,先看哈希值 
    // 哈希值一样 再看equals方法
    // 哈希值不一样 就不看equals
    @Override
    public int hashCode() {
        // this.name.hashcode + age
        // 写这么复杂 就是为了让hashcode减少重复
        // 少调用equals方法 提高存储效率
        final int prime = 31; 
        int result = 1;
        result = prime * result + age;
        result = prime * result + ((name == null) ? 0 : name.hashCode());
        return result;
    }

    // 重写equals方法
    @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 != other.age) // 判断年龄 不同
            return false; //说明不是一个对象
        if (name == null) { // 判断名字是否为空
            if (other.name != null) // 名字一个为空一个不为空
                return false; // 说明不是一个对象
        } else if (!name.equals(other.name)) 
            return false; // 名字不同 则不是一个对象
        return true; // 前面都没走 直接到这里 就说明两个对象时同一个对象
    }   
}
HashSet<Person> hashSet = new HashSet<>();
hashSet.add(new Person("p1",19));
hashSet.add(new Person("p1",19));
hashSet.add(new Person("p2",21));
hashSet.add(new Person("p2",21));
hashSet.add(new Person("p3",22));
hashSet.add(new Person("p3",22));
for (Person person : hashSet) {
    System.out.println(person);
}
System.out.println(hashSet);

输出结果:
这里写图片描述
由此,可以看出:
你要是想使用hashSet去除重复的对象的时候
你需要重写其中的HashCode()和equals()方法

二.TreeSet

1.TreeSet可以用来排序时候默认升序排的,,但是只能排常量的值
代码举例:

TreeSet<Integer> set = new TreeSet<>();
    set.add(1);
    set.add(4);
    set.add(2);
    set.add(3);
    System.out.println(set);

输出结果:
这里写图片描述

2.创建一个treeset 添加4个人 查看是否排序

TreeSet<Person> set = new TreeSet<>();
set.add(new Person("p10",18));
set.add(new Person("p2222",19));
set.add(new Person("p3",20));
set.add(new Person("p422",21));
set.add(new Person("p3",19));
System.out.println(set);

输出结果:你会发现报错
这里写图片描述
需要在Person中重写compareTo方法

public class Person implements Comparable<Person> {
    private String name;
    private int age;

    // 构造方法
    public Person() {

    }

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

    // set/get方法
    public String getName() {
        return name;
    }

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

    public int getAge() {
        return age;
    }

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

    // toString方法的重写
    @Override
    public String toString() {
        return "Person [name=" + name + ", age=" + age + "]";
    }

    // 重写compareTo方法
    // 重写排序规则
    // 主要看字符串长度
    // 次要看姓名的比较
    // 再次要看年龄的比较
    // 如果都相同,就认定是同一个对象
    @Override
    public int compareTo(Person o) {
        int length = this.name.length() - o.getName().length();
        int num2 = this.name.compareTo(o.getName());
        int num3 = this.age - o.getAge();
        return length == 0 ? (num2 == 0 ? num3 : num2) : length;
    }

输出结果:
这里写图片描述
如果TreeSet想向里面存储元素
就看compareTo方法里的返回值到底是正数,0,还是负数:
返回0 –> 只有一个值
返回一个正数 –> 按存进去的顺序 输出
返回一个负数 –> 按存进去的顺序逆向 输出

3.比较器
需求:集合中保存字符串 按字符串长度排序

需要创建一个比较器用来试下你接口中的方法
比较器相当于TreeSet的存储规则

public class ComparatorImpl implements Comparator<String>{
    // 重写排序规则
    // 主要看字符串长度
    // 次要按照字符比
    @Override
    public static void compareTo(String o1,String o2){
        int length = o1.length() - o2.length();                     
        int num = o1.compareTo(o2);
        // 如果有两个字符串两个都相同 但也需要保留
        // 所以返回1 和 -1 都没区别 只要被返回 0 就行
        return length == 0 ? (num == 0 ? 1 : num) : length;
    }
}
// 初始化TreeSet的同时把比较器 传入TreeSet中
TreeSet<String> set = new TreeSet<>(new ComparatorImpl());
set.add("aa");
set.add("zz");
set.add("sb");
set.add("www");
set.add("lanou");
set.add("ly");
set.add("zz");
    for (String string : set) {
        System.out.print(string + " ");
    }

输出结果:
这里写图片描述

4.问题举例
①.在一个集合中存储了无序并且重复的字符串,排序,而且还不能去除重复(用比较器)

ArrayList<String> arrayList = new ArrayList<>();
arrayList.add("aa");
arrayList.add("zz");
arrayList.add("sb");
arrayList.add("www");
arrayList.add("lanou");
arrayList.add("ly");
arrayList.add("zz");

TreeSet<String> treeSet = new TreeSet<>(new ComparatorImpl());
// 把arrayList 整个塞进 treeSet里
treeSet.addAll(arrayList);
// 清空arrayList
arrayList.clear();
// 再把treeSet 整个怼进 arrayList里
arrayList.addAll(treeSet);
System.out.println(arrayList);

这里的比较器也可以用3中的比较器
输出结果:
这里写图片描述
②.键盘输入一个字符串,对其中所有字符进行排序.(不去重复的)

// 键盘接收字符串
System.out.println("请输入一个字符串:");
Scanner scanner = new Scanner(System.in);
// 接收字符串
String string = scanner.nextLine();
// 转成字符数组
char[] array = string.toCharArray();
// 创建TreeSet
TreeSet<Character> set = new TreeSet<>(new charImpl());
    // 把字符数组中的每一个元素 都放进set中
    for (int i = 0; i < array.length; i++) {
        set.add(array[i]);
    }
System.out.println(set);

这里需要写一个比较器:
直接比较两个字符的ASCII码值就好了

public class charImpl implements Comparator<Character> {
    // 重写排序的方法
    @Override
    public int compare(Character o1, Character o2) {
        int num = o1 - o2;
        return num == 0 ? 1 : num;
    }
}

输出结果:
这里写图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值