JAVA集合框架(三)

JAVA集合框架(三)

三. 泛型和工具类

1. 泛型概念

Java泛型是JDK1.5中引入的一个新特性,其本质是参数化类型,把类型作为参数传递。

  • 常见形式有泛型类、泛型接口、泛型方法。

语法:

  • <T,…>T称为类型占位符,表示一种引用类型。

好处:

  • (1)提高代码的重用性
  • (2)防止类型转换异常,提高代码的安全性

2. 泛型类

实例:

public class Application<T> {
    public static void main(String[] args) {
        //使用泛型类来创建对象
        //注意:泛型只能使用引用类型,不同泛型对象之间不能相互赋值
        MyGeneric<String> myGeneric = new MyGeneric<String>();
        myGeneric.t="hello";
        myGeneric.show("Hello World!");
        System.out.println(myGeneric.getT());

        MyGeneric<Integer> myGeneric1 = new MyGeneric<>();
        myGeneric1.t=200;
        myGeneric1.show(100);
        System.out.println(myGeneric1.getT());
    }
}
---------------结果---------------
Hello World!
hello
100
200
public class MyGeneric<T> {
    //T是类型占位符,表示一种引用类型,如果编写多个用逗号隔开
    //使用泛型
    // 1.来创建对象
    T t;
    //2.添加方法
    public void show(T t){
        System.out.println(t);
    }
    //3.泛型作为方法的返回值
    public T getT(){
        return t;
    }
}

3. 泛型接口

实例:

import com.mnm.collection.MyInterfaceImpl;
import com.mnm.collection.MyInterfaceImpl2;

public class Application<T> {
    public static void main(String[] args) {
        MyInterfaceImpl impl = new MyInterfaceImpl();
        impl.sever("abcde");
        MyInterfaceImpl2<Integer> impl2 = new MyInterfaceImpl2<>();
        impl2.sever(20);
    }
}
---------------结果---------------
abcde
20
public interface  MyInterface<T> {
    //泛型接口
    //注意,不能泛型静态常量
    String name="张三";
    T sever(T t);
}
public  class MyInterfaceImpl implements MyInterface<String>{
    @Override
    public String sever(String s) {
        System.out.println(s);
        return null;
    }
}
public class MyInterfaceImpl2<T> implements MyInterface<T> {
    @Override
    public T sever(T t) {
        System.out.println(t);
        return null;
    }
}

4. 泛型方法

实例:

import com.mnm.collection.MyGenericMethod;
public class Application<T> {
    public static void main(String[] args) {
        MyGenericMethod method = new MyGenericMethod();
        method.show();
        System.out.println(method.get("aaa"));
        System.out.println(method.get(100));
    }
}
---------------结果---------------
泛型方法
aaa
100
public class MyGenericMethod {
    //泛型方法
    //语法:<T>返回值类型
    public <T> void show(){
        System.out.println("泛型方法");
    }
    public <T> T get(T t){
        return t;
    }
}

5. 泛型集合

概念

  • 参数化类型、类型安全的集合,强制集合元素的类型必须一致。

特点:

  • 编译时即可检查,而非运行时抛出异常。访问时,不必类型转(拆箱)。
  • 不同泛型之间引用不能相互赋值,泛型不存在多态
import com.mnm.collection.Student;
import java.util.ArrayList;
import java.util.Iterator;

public class Application<T> {
    public static void main(String[] args) {
        //泛型集合
        ArrayList<String> s1 = new ArrayList<>();
        s1.add("aaa");
        s1.add("bbb");
        s1.add("ccc");
        for (String s : s1) {
            System.out.println(s);
        }
        ArrayList<Student> s2 = new ArrayList<>();
        Student student1 = new Student("aaa", 20);
        Student student2 = new Student("aaa", 20);
        Student student3 = new Student("aaa", 20);
        s2.add(student1);
        s2.add(student2);
        s2.add(student3);
        Iterator<Student> iterator = s2.iterator();
        while (iterator.hasNext()){
            System.out.println(iterator.next());
        }
    }
}
---------------结果---------------
aaa
bbb
ccc
[name=aaa, age=20]
[name=aaa, age=20]
[name=aaa, age=20]
public class Student {
    private String name;
    private int age;

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

    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;
    }

    @Override
    public String toString() {
        return "[name="+name+", age="+age+"]";
    }

    @Override
    public boolean equals(Object obj) {
        //是否是同一个对象
        if(this==obj){
            return true;
        }
        //是否是同为空
        if(obj==null){
            return false;
        }
        //判断是否是Student类
        if(obj instanceof Student){
            Student s =(Student)obj;
            //比较属性
            if(this.name.equals(s.getName()) && this.age==s.getAge()){
                return true;
            }
        }
        //不满足,返回false
        return false;
    }
}

四. Set接口和实现类

1. Set子接口

  • 特点:无序、无下标、元素不可重复。
  • 方法:全部继承自Collection中的方法。

2. Set接口的使用

以HashSet为实例:

import java.util.HashSet;
import java.util.Iterator;
import java.util.Set;

public class Application {
    public static void main(String[] args) {
        //set接口的使用
        //特点:无序,无下标,元素不可重复
        //创建集合
        Set<String> set = new HashSet<>();
        //添加元素
        set.add("张三");
        set.add("李四");
        set.add("王五");
        set.add("王五");//重复添加不进去
        System.out.println("元素个数:"+set.size());
        System.out.println(set);
        //删除元素
        set.remove("王五");
        System.out.println(set);
        //遍历
        System.out.println("-----增强for循环-----");
        for (String s : set) {
            System.out.println(s);
        }
        System.out.println("-----迭代器-----");
        Iterator<String> iterator = set.iterator();
        while (iterator.hasNext()){
            System.out.println(iterator.next());
        }
        //判断
        System.out.println("包含李四吗:"+set.contains("李四"));
        System.out.println("是否为空:"+set.isEmpty());
    }
}
---------------结果---------------
元素个数:3
[李四, 张三, 王五]
[李四, 张三]
-----增强for循环-----
李四
张三
-----迭代器-----
李四
张三
包含李四吗:true
是否为空:false

3. Set实现类

HashSet[重点]
  • 基于HashCode实现元素不重复。
  • 当存入元素的哈希码相同时,会调用equals进行确认,如结果为true,则拒绝后者存入。
import java.util.HashSet;
import java.util.Iterator;

public class Application {
    public static void main(String[] args) {
        //HashSet集合的使用
        //存储结构:哈希表(数组+列表+红黑树)
        HashSet<Person> hashSet = new HashSet<>();
        Person p1 = new Person("张三",20);
        Person p2 = new Person("李四",19);
        Person p3 = new Person("王五",18);
        hashSet.add(p1);
        hashSet.add(p2);
        hashSet.add(p3);
        hashSet.add(new Person("王五",18));
        System.out.println("元素个数为:"+hashSet.size());
        System.out.println(hashSet);
    }
}
---------------结果---------------
元素个数为:4
[[name=李四 age=19], [name=王五 age=18], [name=王五 age=18], [name=张三 age=20]]

这时就要讲,hashset的存储方式:

  • (1)根据hashcode计算保存的位置,如果此位置为空,则直接保存,不为空则执行(2)
  • (2)在执行equals方法,如果equals为true,则认为是重复,否则形成链表

重写hashCode和equals后

import java.util.HashSet;
import java.util.Iterator;

public class Application {
    public static void main(String[] args) {
        //HashSet集合的使用
        //存储结构:哈希表(数组+列表+红黑树)
        HashSet<Person> hashSet = new HashSet<>();
        Person p1 = new Person("张三",20);
        Person p2 = new Person("李四",19);
        Person p3 = new Person("王五",18);
        hashSet.add(p1);
        hashSet.add(p2);
        hashSet.add(p3);
        hashSet.add(new Person("王五",18));
        System.out.println("元素个数为:"+hashSet.size());
        System.out.println(hashSet);
    }
}
---------------结果---------------
元素个数为:3
[[name=王五 age=18], [name=张三 age=20], [name=李四 age=19]]

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

    public Person() {
    }

    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;
    }

    @Override
    public String toString() {
        return "[name="+name+" age="+age+"]";
    }

   @Override
    public int hashCode() {
        int n1=this.name.hashCode();
        int n2=this.age;
        return n1+n2;
    }
    @Override
    public boolean equals(Object obj) {
        //是否是同一个对象
        if(this==obj){
            return true;
        }
        //是否是同为空
        if(obj==null){
            return false;
        }
        //判断是否是Person类
        if(obj instanceof Person){
            Person s =(Person) obj;
            //比较属性
            if(this.name.equals(s.getName()) && this.age==s.getAge()){
                return true;
            }
        }
        //不满足,返回false
        return false;
    }
}

IDEA中可快捷输入:Alt+Insert键
在这里插入图片描述
系统重写的代码

@Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (!(o instanceof Person)) return false;
        Person person = (Person) o;
        return getAge() == person.getAge() && getName().equals(person.getName());
    }

    @Override
    public int hashCode() {
        return Objects.hash(getName(), getAge());
    }

小知识:
在hashCode的源码中,可以看到用到数字31,为什么用31呢?

public static int hashCode(Object a[]) {
        if (a == null)
            return 0;
        int result = 1;
        for (Object element : a)
            result = 31 * result + (element == null ? 0 : element.hashCode());
        return result;
    }

原因:

  • (1). 31是一个质数,可以减少散列冲突(数据结构课程中应该讲过,没有学过的可以了解一下散列表)
  • (2). 31可以提高运行效率,31*i=(i<<5)-i,如i=1时,31=100000B-1

如有不同见解,可以写在评论区

TreeSet
  • 基于排列顺序实现元素不重复。
  • 实现了SortedSet接口,对集合元素自动排序。
  • 元素对象的类型必须实现Comparable接口,指定排序规则。
  • 通过CompareTo方法确定是否为重复元素。

TreeSet采用红黑树来排列,红黑树是平衡二叉排序树加上一些颜色来区分(也是数据结构课程中的东西,可以自行搜索视频观看)

实例一:
import java.util.Iterator;
import java.util.TreeSet;

public class Application {
    public static void main(String[] args) {
        //TreeSet的使用  存储结构:红黑树
        TreeSet<String> treeSet = new TreeSet<>();
        //添加元素
        treeSet.add("def");
        treeSet.add("ghi");
        treeSet.add("abc");
        System.out.println("元素个数为:"+treeSet.size());
        System.out.println(treeSet);
        //删除
        treeSet.remove("def");
        System.out.println(treeSet);
        //遍历
        System.out.println("-----增强for-----");
        for (String s : treeSet) {
            System.out.println(s);
        }
        System.out.println("-----迭代器-----");
        Iterator<String> iterator = treeSet.iterator();
        while (iterator.hasNext()){
            System.out.println(iterator.next());
        }
        //判断
        System.out.println("是否包含def:"+treeSet.contains("def"));
        System.out.println("是否为空:"+treeSet.isEmpty());
    }
}
---------------结果---------------
元素个数为:3
[abc, def, ghi]
[abc, ghi]
-----增强for-----
abc
ghi
-----迭代器-----
abc
ghi
是否包含def:false
是否为空:false
实例二:
import java.util.Iterator;
import java.util.TreeSet;

public class Application {
    public static void main(String[] args) {
        //TreeSet的使用  存储结构:红黑树
        //要求:元素必须实现Comparable接口,compareTo方法返回值为0,认为是重复元素
        TreeSet<Person> treeSet = new TreeSet<>();
        //添加元素
        Person p1 = new Person("a", 20);
        Person p2 = new Person("b", 19);
        Person p3 = new Person("c", 18);
        Person p4 = new Person("d", 18);
        treeSet.add(p1);
        treeSet.add(p2);
        treeSet.add(p3);
        treeSet.add(p4);
        System.out.println("元素个数为:"+treeSet.size());
        System.out.println(treeSet);
        //删除
        treeSet.remove(p4);
        System.out.println("删除后:"+treeSet);
        //遍历
        System.out.println("-----增强for-----");
        for (Person person : treeSet) {
            System.out.println(person);
        }
        System.out.println("-----迭代器-----");
        Iterator<Person> iterator = treeSet.iterator();
        while (iterator.hasNext()){
            System.out.println(iterator.next());
        }
        //判断
        System.out.println("包含p3吗:"+treeSet.contains(p3));
        System.out.println("是否为空:"+treeSet.isEmpty());
    }
}
---------------结果---------------
元素个数为:4
[[name=a age=20], [name=b age=19], [name=c age=18], [name=d age=18]]
删除后:[[name=a age=20], [name=b age=19], [name=c age=18]]
-----增强for-----
[name=a age=20]
[name=b age=19]
[name=c age=18]
-----迭代器-----
[name=a age=20]
[name=b age=19]
[name=c age=18]
包含p3吗:true
是否为空:false
import java.util.Objects;

public class Person implements Comparable<Person>{
    private String name;
    private int age;
    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public Person() {
    }

    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;
    }

    @Override
    public String toString() {
        return "[name="+name+" age="+age+"]";
    }

    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (!(o instanceof Person)) return false;
        Person person = (Person) o;
        return getAge() == person.getAge() && getName().equals(person.getName());
    }

    @Override
    public int hashCode() {
        return Objects.hash(getName(), getAge());
    }

    @Override
    public int compareTo(Person o) {
        int n1=this.getName().compareTo(o.getName());
        int n2=this.age-o.getAge();
        //先按姓名比较,然后按年龄比较
        return n1==0?n2:n1;
    }
}
Comparator接口

实例:

import java.util.Comparator;
import java.util.TreeSet;

public class Application {
    public static void main(String[] args) {
        //TreeSet的使用
        //Comparator:实现定制比较(比较器)
        //Comparable:可比较的
        //创建集合,并指定了比较规则
        TreeSet<Person> treeSet= new TreeSet<>(new Comparator<Person>() {
            @Override
            public int compare(Person o1, Person o2) {
                int n1=o1.getAge()-o2.getAge();
                int n2=o1.getName().compareTo(o2.getName());
                return n1==0?n2:n1;
            }
        });
        Person p1 = new Person("a", 20);
        Person p2 = new Person("b", 19);
        Person p3 = new Person("c", 18);
        treeSet.add(p1);
        treeSet.add(p2);
        treeSet.add(p3);
        System.out.println("元素个数:"+treeSet.size());
        System.out.println(treeSet);
    }
}
---------------结果---------------
元素个数:3
[[name=c age=18], [name=b age=19], [name=a age=20]]
案例:

要求:

  • 使用TreeSet集合实现字符串按照长度排序,长度相同按照字典表顺序排序
  • 字符串为:HelloWorld,zhangsan,lisi,wangwu,beijing,xian,nanjing
import java.util.Comparator;
import java.util.TreeSet;

public class Application {
    public static void main(String[] args) {
        TreeSet<String> str = new TreeSet<>(new Comparator<String>() {
            @Override
            public int compare(String o1, String o2) {
                int n1=o1.length()-o2.length();
                int n2=o1.compareTo(o2);
                return n1==0?n2:n1;
            }
        });
        str.add("HelloWorld");
        str.add("zhangsan");
        str.add("lisi");
        str.add("wangwu");
        str.add("beijing");
        str.add("xian");
        str.add("nanjing");
        System.out.println("元素个数为:"+str.size());
        System.out.println(str);
    }
}
---------------结果---------------
元素个数为:7
[lisi, xian, wangwu, beijing, nanjing, zhangsan, HelloWorld]

另几篇链接:
JAVA集合框架(一)链接
JAVA集合框架(二)链接
JAVA集合框架(四)链接

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值