• 集合和泛型


    目录

    1.1 集合框架

    1.1.1为什么要使用集合框架

    1.1.2单值集合的顶层接口:Collection 家族

    1.1.3键值映射集合的顶层接口:Map 家族

    1.2 Set 接口

    1.2.1HashSet 类的使用

    1.2.2HashSet 类无重复特性体现

    总结: 

    1.2.3TreeSet 类的使用

    1.2.3.1使用 TreeSet 生成数组

    1.2.4内部比较器 Comparable 接口

     1.3 List 接口

    1.3.1 LinkedList 与 Comparator 结合使用

    1.4 泛型

    1.4.1泛型的基本使用

    1.5 Iterator 迭代器

    1.5.1Iterator 接口的使用

    1.5.2使用 Iterator 显示数据信息

    1.6 Map 接口

    1.6.1HashMap 类的使用

    1.6.2 Map 映射数据的遍历

    1.7工具类

    1.7.1Collections 工具类的使用 

    1.7.2 Arrays 工具类的使用

    综合案例


    1.1集合框架

    1.1.1为什么要使用集合框架

            之前我们学习过数组,知道了数组可以存储多个数据类型相同的元素,但面对频繁增加、删除、修改元素的要求以及动态扩容要求时显得捉襟见肘。为此,JDK 提供了一套 “集合” 框架,这套框架是对常见数据结构的实现,不仅可存储数据,还提供了丰富的访问和处理数据的操作。在面向对象思想里,数据结构也被认为是一个容器,所以集合、容器等词汇经常被交替使用。

            Java 集合框架支持两种类型的容器:一种是为了存储一个元素集合,简称为集合(collection);另一种是为了存储键/值对,称为映射(map,或称图)。

            使用数组存放多个  对象的信息会存在很多问题,首先,Java 语言中的数组长度是固定的,旦创建出指定长度的数组以后,就给内存分配了相应的存储空间如果数组长度设置大了,又会造成空间浪费,删除数组元素全部要前移一位,但这种元素的移动是比较消耗系统资源的。

    1.1.2单值集合的顶层接口:Collection 家族

            集合框架可以分为 Collection 和 Map 两类,现在开始学习 Collection。Collection 是一个顶层接口,一些 Collection 接口的实现类允许有重复的元素,而另一些则不允许;一些 Collection 是有序的,而另一些则是无序的。

            JDK 不提供 Collection 接口的任何直接实现类,而是提供了更具体的子接口,如 Set 接口和 List 接口。这些子接口继承 Collection 接口的方法,然后再对 Collection 接口从不同角度进行重写或扩充。

    Collection 接口主要有三个子接口,分别是 Set 接口、List 接口和 Queue 接口,下面简要介绍这三个接口。

    • Set 接口

      Set 实例用于存储一组不重复的,无序的元素。

    • List 接口

      List 实例是一个有序集合。程序员可对 List 中每个元素的位置进行精确控制,可以根据索引来访问元素,此外 List 中的元素是可以重复的。

    • Queue 接口

      Queue 中的元素遵循先进先出的规则,是对数据结构 “队列” 的实现。

    1.1.3键值映射集合的顶层接口:Map 家族

            Map 接口定义了存储和操作一组 “键(key)值(value)” 映射对的方法。

            Map 接口和 Collection 接口的本质区别在于,Collection 接口里存放的是一系列单值对象,而 Map 接口里存放的是一系列 key-value 对象。Map 中的 key 不能重复,每个 key 最多只能映射到一个值。HashMap ,Hashtable 和TreeMap 是 Map 接口的实现类。

    1.2Set 接口

    1.2.1HashSet 类的使用

            Set 接口是 Collection 的子接口。Set 接口中的元素是不能重复的、无序的,这里的 “无序” 是指向 Set 中输入的元素,与从 Set 中输出元素的顺序是不一致的。

            例如,向 Set 接口中依次增加 “北京”、“深圳” 和 “西安” 三个元素,但输出顺序却是 “西安”、“北京”和“深圳” 。对于开发者而言,只需要了解这一 “无序” 的特性即可,不必深究其原因。

    下面列出了 Set 接口继承自 Collection 接口的主要方法。

    • boolean add(Object obj)

      向集合中添加一个 obj 元素,并且 obj 不能和集合中现有数据元素重复,添加成功后返回 true。如果添加的是重复元素,则添加操作无效,并返回 false。

    • void clear()

      移除此集合中的所有数据元素,即将集合清空。

    • boolean contains(Object obj)

      判断此集合中是否包含 obj,如果包含,则返回 true。

    • boolean isEmpty()

      判断集合是否为空,为空则返回 true。

    • Iterator iterator()

      返回一个 Iterator 对象,可用它来遍历集合中的数据元素。

    • boolean remove(Object obj)

      如果此集合中包含 obj,则将其删除,并返回 true。

    • int size()

      返回集合中真实存放数据元素的个数,注意与数组、字符串获取长度的方法的区别。

    • Object[] toArray()

      返回一个数组,该数组包含集合中的所有数据元素。

    本实验主要学习了 Set 接口和 HashSet 类的使用:

    1. Set 接口的特性:无重复,无序。
    2. Set 接口下继承父接口的方法。
    3. Set 接口的实现类 HashSet 的使用:
      • 通过 HashSet 无参构造器创建 Set 对象。
      • 使用了 add()、contains()、remove()、size() 等常用的方法对集合元素进行操作。
    1. import java.util.HashSet;
    2. import java.util.Set;
    3. /**
    4. * HashSet 类的使用
    5. */
    6. public class TestSet {
    7. public static void main(String[] args) {
    8. //创建一个HashSet对象,存放学生姓名信息
    9. Set nameSet = new HashSet();
    10. nameSet.add("王云");
    11. nameSet.add("刘静涛");
    12. nameSet.add("南天华");
    13. nameSet.add("雷静");
    14. nameSet.add("王云"); //增加已有的数据元素
    15. System.out.println("再次添加王云是否成功:" + nameSet.add("王云"));
    16. System.out.println("显示集合内容:" + nameSet);
    17. System.out.println("集合里是否包含南天华:" + nameSet.contains("南天华"));
    18. System.out.println("从集合中删除\"南天华\"...");
    19. nameSet.remove("南天华");
    20. System.out.println("集合里是否包含南天华:" + nameSet.contains("南天华"));
    21. System.out.println("集合中的元素个数为:" + nameSet.size());
    22. }
    23. }

    1.2.2HashSet 类无重复特性体现

            HashSet 实现类的使用还是挺简单的,上一个实验中我们提到了 Set 接口的特性是无序无重复。对吧!

            那么需要思考一个问题:HashSet 是如何判断元素重复的?如果逐个比较 HashSet 中的全部元素,显然是一种效率低下的做法。因此 HashSet 的底层引入了 hashcode。

            hashcode 最初定义在 Object 类中,如果两个对象相等,那么这两个对象的 hashcode 值相同,因此根据逆否定理可知如果两个对象的 hashcode 值不同,那么这两个对象不相等。但反之,如果两个对象的 hashcode 值相同,则这两个对象可能相等,也可能不等,需要再通过 equals() 方法进一步比较这两个对象的内容是否相同。

            当向 HashSet 中增加元素时,HashSet 会先计算此元素的 hashcode,如果 hashcode 值与 HashSet 集合中的其他元素的 hashcode 值都不相同,那么就能断定此元素是唯一的。否则,如果 hashcode 值与 HashSet 集合中的某个元素的 hashcode 值相同,HashSet 就会继续调用 equals() 方法进一步判断它们的内容是否相同,如果相同就忽略这个新增的元素,如果不同就把它增加到 HashSet 中。

            因此,在实际开发中,当使用 HashSet 存放某个自定义对象时,就得先在这个对象的定义类中重写 hashCode() 和 equals() 方法。hashcode 值是对象的映射地址,而 equals() 用于比较两个对象,判断 2 个元素的地址是否相等。在重写时,hashCode() 方法需要自定义 “映射地址” 的映射规则,equals() 方法需要自定义对象的 “比较” 规则。一般而言,映射规则和比较规则都需要借助于对象的所有属性进行计算。

            实际上,如何在 hashCode() 方法中设计计算公式是一个数学问题,我们不必深究。计算公式的目的是为了 “尽可能的避免不同对象计算出的 hash 值相同” ,普通开发者通常只需在 hashCode() 方法中使用全部的属性值进行计算即可。例如,也可以将上面程序中的 hashCode() 方法重写为以下的简易形式。

    1. @Override
    2. public int hashCode() {
    3. return name.hashCode() & oil;
    4. }

    总结: 

            在向 HashSet 集合中增加元素时,会先计算此元素的 hashcode 值,如果 HashSet 集合中没有此 hashcode 值,那么此元素就可以插入。如果 hashcode 值与 HashSet 集合中的某个元素的 hashcode 值相同,HashSet 就会继续调用 euqals() 方法进一步判断它们的内容是否相同,如果相同就忽略这个新增的元素,如果不同才能把它增加到 HashSet 集合中。

    1.2.3TreeSet 类的使用

            TreeSet 类在实现了 Set 接口的同时,也实现了 SortedSet 接口,是一个具有排序功能的 Set 接口实现类。

            TreeSet 集合中的元素默认是按照自然升序排列,并且 TreeSet 集合中的对象需要实现 Comparable 接口。Comparable 接口用于比较集合中各个元素的大小,常用于排序操作。

    1. import java.util.Set;
    2. import java.util.TreeSet;
    3. public class TestTreeSet {
    4. public static void main(String[] args) {
    5. Set ts = new TreeSet();
    6. ts.add("王云");
    7. ts.add("刘静涛");
    8. ts.add("南天华");
    9. System.out.println(ts);
    10. }
    11. }

            从运行结果可以看出,TreeSet 集合 ts 里面的元素不是毫无规律的排序,而是按照自然升序(这里是指 “字典” 里的顺序)进行了排序。这是因为 TreeSet 集合中的元素是 String 类,而 String 类实现了 Comparable 接口。但如果 TreeSet 中的元素不是 String 类,如何进行排序呢?后面 “比较器” 实验中会为大家进行讲解。

    1.2.3.1使用 TreeSet 生成数组

    • toArray() 方法,把集合中的所有数据提取到一个新的数组中。
    1. import java.util.Random;
    2. import java.util.TreeSet;
    3. /**
    4. * 使用 TreeSet 生成数组
    5. */
    6. public class RandomSortArray {
    7. public static void main(String[] args) {
    8. // 创建 TreeSet 对象
    9. TreeSet set = new TreeSet();
    10. // 创建 Random 对象
    11. Random ran =new Random();
    12. int count =0;
    13. while(count<10){
    14. // 提取 0 - 99 的随机数加入到集合中
    15. boolean succeed = set.add(ran.nextInt(100));
    16. if(succeed){
    17. count ++;
    18. }
    19. }
    20. int size = set.size();
    21. // 创建整型数组
    22. Integer[] array = new Integer[size];
    23. // 将集合元素转换为数组元素
    24. set.toArray(array);
    25. System.out.print("生成不重复随机数组内容如下:");
    26. for(int value : array){
    27. System.out.print(value + " ");
    28. }
    29. }
    30. }

    1.2.4内部比较器 Comparable 接口

            JDK 提供了 Comparable 和 Comparator 两个接口,都可以用于定义集合元素的排序规则。如果程序员想定义自己的排序方式,一种简单的方法就是让加入 TreeSet 集合中的对象所属的类实现 Comparable 接口,通过实现 compareTo(Object o) 方法,达到排序的目的。

            假设有这样的需求,学生对象有两个属性,分别是学号和姓名。希望将这些学生对象加入 TreeSet 集合后,按照学号从小到大进行排序,如果学号相同再按照姓名自然排序。来看 Student 类需要实现 Comparable 接      口。

    1. /**
    2. * 学生类实现 Comparable 接口
    3. */
    4. public class Student implements Comparable {
    5. // 学生学号
    6. int stuNum = -1;
    7. // 学生姓名
    8. String stuName = "";
    9. Student(String name, int num) {
    10. this.stuNum = num;
    11. this.stuName = name;
    12. }
    13. // 返回该对象的字符串表示,利于输出
    14. public String toString() {
    15. return "学号为:" + stuNum + " 的学生,姓名为:" + stuName;
    16. }
    17. // 实现 Comparable 的 compareTo() 方法
    18. public int compareTo(Object o) {
    19. Student input = (Student) o;
    20. // 此学生对象的学号和指定学生对象的学号比较
    21. // 此学生对象学号若大则res为1,若小则res为-1,相同的话res = 0
    22. int res = stuNum > input.stuNum ? 1 : (stuNum == input.stuNum ? 0 : -1);
    23. // 若学号相同,则按照String类自然排序比较学生姓名 因为String 类实现了 Comparable 接口
    24. if (res == 0) {
    25. res = stuName.compareTo(input.stuName);
    26. }
    27. return res;
    28. }
    29. }
    1. import java.util.Set;
    2. import java.util.TreeSet;
    3. /**
    4. * 测试类
    5. */
    6. class TestComparable {
    7. public static void main(String[] args) {
    8. //用有序的 TreeSet 存储学生对象
    9. Set stuTS = new TreeSet();
    10. stuTS.add(new Student("王云", 1));
    11. stuTS.add(new Student("南天华", 3));
    12. stuTS.add(new Student("刘静涛", 2));
    13. stuTS.add(new Student("张平", 3));
    14. //循环输出
    15. for(Object stu : stuTS)
    16. System.out.println(stu);
    17. }
    18. }

     1.3 List 接口

    List 是 Collection 接口的子接口,List 中的元素是有序的,而且可以重复。List 集合中的数据元素都对应一个整数形式的序号索引,记录其在集合中的位置,可以根据此序号存取元素。JDK 中常用的 List 实现类是 ArrayList 和 LinkedList。

    List 接口继承自 Collection 接口,除了拥有 Collection 接口所拥有的方法外,还拥有下列方法:

    • void add(int index,Object o)

      在集合的指定 index 位置处,插入指定的 o 元素。

    • Object get(int index)

      返回集合中 index 位置的数据元素。

    • int indexOf(Object o)

      返回此集合中第一次出现的指定 o 元素的索引,如果此集合不包含 o 元素,则返回-1。

    • int lastIndexOf(Object o)

      返回此集合中最后出现的指定 o 元素的索引,如果此集合不包含 o 元素,则返回-1。

    • Object remove(int index)

      移除集合中 index 位置的数据元素。

    • Object set(int index,Object o)

      用指定的 o 元素替换集合中 index 位置的数据元素。

    ArrayList 实现了 List 接口,其底层采用的数据结构是数组。另一个 List 接口的实现类是 LinkedList,它在存储方式上采用链表进行链式存储。

            根据数据结构的知识可知,数组(顺序表)在插入或删除数据元素时,需要批量移动数据元素,故性能较差;但在根据索引获取数据元素时,因为数组是连续存储的,所以在遍历元素或随机访问元素时效率高。

    那么本实验需要学习的 ArrayList 实现类的底层就是数组,因此 ArrayList 实现类更加适合根据索引访问元素的操作。

    假设车辆管理有如下需求:

    1. 用户可以按照车辆入库的顺序查阅车辆信息。
    2. 所有车辆有连续的编号,当用户输入车辆的编号后系统显示车辆完整信息。
    1. /**
    2. * 自定义车辆信息类
    3. */
    4. public class Vehicle {
    5. private String name;
    6. private int oil;
    7. public Vehicle() {
    8. }
    9. public Vehicle(String name, int oil) {
    10. this.name = name;
    11. this.oil = oil;
    12. }
    13. public String getName() {
    14. return name;
    15. }
    16. public void setName(String name) {
    17. this.name = name;
    18. }
    19. public int getOil() {
    20. return oil;
    21. }
    22. public void setOil(int oil) {
    23. this.oil = oil;
    24. }
    25. }
    1. import java.util.ArrayList;
    2. import java.util.List;
    3. import java.util.Scanner;
    4. /**
    5. * ArrayList 类的基本使用
    6. */
    7. public class TestArrayList{
    8. public static void main(String[] args) {
    9. Scanner input = new Scanner(System.in);
    10. // 创建 ArrayList 集合,用于存放车辆
    11. List vehAL = new ArrayList();
    12. Vehicle c1 = new Car("战神","长城");
    13. Vehicle c2 = new Car("跑得快","红旗");
    14. Vehicle t1 = new Truck("大力士","5吨");
    15. Vehicle t2 = new Truck("大力士二代","10吨");
    16. // 将 c1 添加到 vehAL 集合的末尾
    17. vehAL.add(c1);
    18. vehAL.add(c2);
    19. vehAL.add(t1);
    20. vehAL.add(t2);
    21. System.out.println("*** 显示全部车辆 ***");
    22. // 用于显示序号
    23. int num = 1;
    24. // 增强for循环遍历
    25. for(Object obj:vehAL){
    26. if(obj instanceof Car) {
    27. Car car = (Car)obj;
    28. System.out.println(num + " 该车是轿车,其车名为:" + car.getName());
    29. }else{
    30. Truck truck = (Truck)obj;
    31. System.out.println(num + " 该车是卡车,其车名为:" + truck.getName());
    32. }
    33. num++;
    34. }
    35. System.out.print("请输入要显示车名的车辆编号:");
    36. String name = ((Vehicle)vehAL.get(input.nextInt()-1)).getName();
    37. System.out.println("车辆名称为:"+name);
    38. }
    39. }
    40. //轿车类
    41. class Car extends Vehicle{
    42. //品牌
    43. private String brand = "红旗";
    44. //构造方法,指定车名和品牌
    45. public Car(String name, String brand) {
    46. super(name, 20);
    47. this.brand = brand;
    48. }
    49. //获取品牌
    50. public String getBrand() {
    51. return brand;
    52. }
    53. }
    54. //卡车类
    55. class Truck extends Vehicle{
    56. // 吨位
    57. private String load = "10吨";
    58. //构造方法,指定车名和品牌
    59. public Truck(String name, String load) {
    60. super(name, 20);
    61. this.load = load;
    62. }
    63. //获取吨位
    64. public String getLoad() {
    65. return load;
    66. }
    67. }

    1.3.1 LinkedList 与 Comparator 结合使用

    本实验将学习 LinkedList 与 Comparator 结合使用。LinkedList,它在存储方式上采用链表进行链式存储;而 Comparator 是外部比较器。

    LinkedList 的底层是链表。LinkedList 和 ArrayList 在应用层面类似,只是底层存储结构上的差异导致了二者对于不同操作,存在性能上的差异。这其实就是顺序表和链表之间的差异。一般而言,对于 “索引访问” 较多的集合操作建议使用 ArrayList 实现类,而对于 “增删” 较多的集合操作建议使用 LinkedList 实现类。

    LinkedList 实现类除了拥有 ArrayList 实现类提供的方法外,还增加了如下一些方法:

    • void addFirst(Object o)

      将指定数据元素插入此集合的开头。

    • void addLast(Object o)

      将指定数据元素插入此集合的结尾。

    • Object getFirst()

      返回此集合的第一个数据元素。

    • Object getLast()

      返回此集合的最后一个数据元素。

    • Object removeFirst()

      移除并返回此集合的第一个数据元素。

    • Object removeLast()

      移除并返回此集合的最后一个数据元素。

            Comparator 可以理解为一个专用的比较器,当集合中的对象不支持自比较或者自比较的功能不能满足程序员的需求时,就可以写一个实现 Comparator 接口的比较器来完成两个对象之间的比较,从而实现按比较器规则进行排序的功能。

            例如,要比较的对象是 JDK 中内置的某个类,而这个类又没有实现 Comparable 接口,因此我们是无法直接修改 JDK 内置类的源码的,因此就不能通过重写 compareTo(Object o) 方法来定义排序规则了,而应该使用 Comparator 接口实现比较器功能。

            接下来,在外部定义一个姓名比较器和一个学号比较器,然后在使用 Collections 工具类的 sort(List list, Comparator c) 方法时选择使用其中一种外部比较器,对集合里的学生信息按姓名、学号分别排序输出。

    1. import java.util.Collections;
    2. import java.util.Comparator;
    3. import java.util.LinkedList;
    4. /**
    5. * LinkedList 与 Comparator 结合使用
    6. */
    7. public class TestLinkedList{
    8. public static void main(String[] args){
    9. //用LinkedList存储学生对象
    10. LinkedList stuLL = new LinkedList();
    11. stuLL.add(new Student("王云",1));
    12. stuLL.add(new Student("南天华",3));
    13. stuLL.add(new Student("刘静涛",2));
    14. //使用sort方法,按姓名比较器进行排序
    15. Collections.sort(stuLL,new NameComparator());
    16. System.out.println("*** 按学生姓名顺序输出学生信息 ***");
    17. for (Object object : stuLL) {
    18. System.out.println(object);
    19. }
    20. //使用sort方法,按学号比较器进行排序
    21. Collections.sort(stuLL,new NumComparator());
    22. System.out.println("*** 按学生学号顺序输出学生信息 ***");
    23. for (Object object : stuLL) {
    24. System.out.println(object);
    25. }
    26. }
    27. }
    28. // 定义学生对象,未实现Comparable接口
    29. class Student{
    30. int stuNum = -1;
    31. String stuName = "";
    32. Student(String name, int num) {
    33. this.stuNum = num;
    34. this.stuName = name;
    35. }
    36. @Override
    37. public String toString() {
    38. return "学号为:" + stuNum + " 的学生,姓名为:" + stuName;
    39. }
    40. }
    41. //定义一个姓名比较器
    42. class NameComparator implements Comparator {
    43. //实现Comparator接口的compare()方法
    44. public int compare(Object op1, Object op2) {
    45. Student eOp1 = (Student)op1;
    46. Student eOp2 = (Student)op2;
    47. //通过调用String类的compareTo()方法进行比较
    48. return eOp1.stuName.compareTo(eOp2.stuName);
    49. }
    50. }
    51. //定义一个学号比较器
    52. class NumComparator implements Comparator {
    53. //实现Comparator接口的compare()方法
    54. public int compare(Object op1, Object op2) {
    55. Student eOp1 = (Student)op1;
    56. Student eOp2 = (Student)op2;
    57. return eOp1.stuNum - eOp2.stuNum;
    58. }
    59. }

    1.4泛型

    1.4.1泛型的基本使用

            首先说明,在初级阶段,为了快速的掌握泛型的用法,本节只讲解 “使用泛型限制集合元素类型” 这一个核心内容,其余较为深入的泛型知识暂不做介绍。

            在之前使用集合的时候,装入集合的各种类型的元素都被当作 Object 对待,而非元素自身的类型。因此从集合中取出某个元素时,就需要进行类型转换,这种做法效率低下且容易出错。

            🤔 那么如何解决这个问题呢?可以使用泛型。

            泛型是指在定义集合的同时也定义集合中元素的类型,需要 “< >” 进行指定,其语法形式如下:

    集合<数据类型> 引用名 = new 集合实现类<数据类型> ();

    ⭐ 注意:使用泛型约束的数据类型必须是对象类型,而不能是基本数据类型。

    以下代码就限制了 List 集合中只能存放 String 类型的元素。

    List list = new ArrayList();

    在 JDK1.7 之后,= 右边 < > 中的的 String 等类型也可以省略,也可以写成以下的等价形式:

    List list = new ArrayList<>();

            在定义集合的同时使用泛型,用 “< >” 进行指定集合中元素的类型后,再从集合中取出某个元素时,就不需要进行类型转换,不仅可以提高程序的效率,也让程序更加清晰明了,易于理解。

    1.5 Iterator 迭代器

    1.5.1Iterator 接口的使用

    前面学习的 Collection 接口、Set 接口和 List 接口,它们的实现类都没有提供遍历集合元素的方法。

    Iterator 接口为遍历集合而生,是 Java 语言解决集合遍历的一个工具。

    iterator() 方法定义在 Collection 接口中,因此所有单值集合的实现类,都可以通过 iterator() 方法实现遍历。iterator() 方法返回值是 Iterator 对象,通过 Iterator 接口的 hasNext() 和 next() 方法即可实现对集合元素的遍历。

    下面是 Iterator 接口的三个方法:

    • boolean hasNext()

      判断是否存在下一个可访问的数据元素。

    • Object next()

      返回要访问的下一个数据元素,通常和 hasNext() 在一起使用。

    • void remove()

      从迭代器指向的 Collection 集合中移除迭代器返回的上一个数据元素。

    1. import java.util.HashSet;
    2. import java.util.Iterator;
    3. import java.util.Set;
    4. /**
    5. * 使用迭代器遍历集合数据
    6. */
    7. public class TestIterator {
    8. public static void main(String[] args) {
    9. // 创建 HashSet 集合,用于存放车辆
    10. Set vehSet = new HashSet<>();
    11. // 创建两个轿车对象、两个卡车对象,并加入 HashSet 集合中
    12. Vehicle c1 = new Car("战神", "长城");
    13. Vehicle c2 = new Car("跑得快", "红旗");
    14. Vehicle t1 = new Truck("大力士", "5吨");
    15. Vehicle t2 = new Truck("大力士二代", "10吨");
    16. vehSet.add(c1);
    17. vehSet.add(c2);
    18. vehSet.add(t1);
    19. vehSet.add(t2);
    20. // 使用迭代器循环输出
    21. Iterator it = vehSet.iterator();
    22. System.out.println("*** 显示集合中元素信息 ***");
    23. while (it.hasNext()) {
    24. Vehicle vehicle = it.next();
    25. if (vehicle instanceof Car) {
    26. Car car = (Car) vehicle;
    27. //调用 Car 类的特有方法 getBrand()
    28. System.out.println("该车是轿车,其品牌为:" + car.getBrand());
    29. } else {
    30. Truck truck = (Truck) vehicle;
    31. //调用 Truck 类的特有方法 getLoad()
    32. System.out.println("该车是卡车,其吨位为:" + truck.getLoad());
    33. }
    34. System.out.println("车辆名称:" + vehicle.getName());
    35. // 集合元素分隔显式
    36. System.out.println("--------------------------");
    37. }
    38. }
    39. }
    1. /**
    2. * 自定义车辆信息类
    3. */
    4. public class Vehicle {
    5. private String name;
    6. private int oil;
    7. public Vehicle() {
    8. }
    9. public Vehicle(String name, int oil) {
    10. this.name = name;
    11. this.oil = oil;
    12. }
    13. public String getName() {
    14. return name;
    15. }
    16. public void setName(String name) {
    17. this.name = name;
    18. }
    19. public int getOil() {
    20. return oil;
    21. }
    22. public void setOil(int oil) {
    23. this.oil = oil;
    24. }
    25. }
    26. //轿车类
    27. class Car extends Vehicle{
    28. //品牌
    29. private String brand = "红旗";
    30. //构造方法,指定车名和品牌
    31. public Car(String name, String brand) {
    32. super(name, 20);
    33. this.brand = brand;
    34. }
    35. //获取品牌
    36. public String getBrand() {
    37. return brand;
    38. }
    39. }
    40. //卡车类
    41. class Truck extends Vehicle{
    42. // 吨位
    43. private String load = "10吨";
    44. //构造方法,指定车名和品牌
    45. public Truck(String name, String load) {
    46. super(name, 20);
    47. this.load = load;
    48. }
    49. //获取吨位
    50. public String getLoad() {
    51. return load;
    52. }
    53. }

    1.5.2使用 Iterator 显示数据信息

    1. import java.util.Iterator;
    2. import java.util.Set;
    3. import java.util.TreeSet;
    4. public class DataShowIterator {
    5. public static void main(String[] args) {
    6. Set treeSet = new TreeSet();
    7. int[] a = {12, 45, 23, 86, 100, 78, 546, 1, 45, 99, 136, 23};
    8. for (int i = 0; i < 12; i++) {
    9. treeSet.add(a[i]);
    10. }
    11. Iterator it = treeSet.iterator();
    12. while (it.hasNext()) {
    13. System.out.println(it.next());
    14. }
    15. }
    16. }

    1.6 Map 接口

    1.6.1HashMap 类的使用

    Map 接口,用于保存具有映射关系的键值对数据。

    Map 接口中的 key 和 value 可以是任何引用类型的数据,key 不允许重复原因和 HashSet 一样,value 可以重复,key 和 value 都可以是 null 值,但需要注意的是,key 为 null 只能有一个,value 为 null 可以多个,它们之间存在单向一对一关系,也就是说通过指定存在的 key 一定找到对应的 value 值。

    Map 接口的常用方法如下:

    • Object put(Object key,Object value)

      将指定键值对(key 和 value)添加到 Map 集合中,如果此 Map 集合以前包含一个该键 key 的键值对,则用参数 key 和 value 替换旧值。

    • Object get(Object key)

      返回指定键 key 所对应的值,如果此 Map 集合中不包含该键 key,则返回 null。

    • Object remove(Object key)

      如果存在指定键 key 的键值对,则将该键值对从此 Map 集合中移除。

    • Set keySet()

      返回此 Map 集合中包含的键的 Set 集合。在上面的程序最后添加下面的语句:System.out. println(domains.keySet());,则会输出[com, edu, org, net]

    • Collection values()

      返回此 Map 集合中包含的值的 Collection 集合。在上面的程序最后添加下面的语句:System.out.println(domains.values());,则会输出[工商企业,教研机构,非营利组织,网络服务商]

    • boolean containsKey(Object key)

      如果此 Map 集合包含指定键 key 的键值对,则返回 true。

    • boolean containsValue(Object value)

      如果此 Map 集合将一个或多个键映射到指定值,则返回 true。

    • int size()

      返回此 Map 集合的键值对的个数。

    Map 接口常用的实现类有 HashMap 和 Hashtable,本实验主要讲 HashMap 类的使用。

    1. import java.util.HashMap;
    2. import java.util.Map;
    3. /**
    4. * HashMap 类的基本使用
    5. */
    6. public class TestHashMap {
    7. public static void main(String[] args) {
    8. // 使用 HashMap 存储域名和含义键值对的集合
    9. Map domains = new HashMap<>();
    10. domains.put("com", "工商企业");
    11. domains.put("net", "网络服务商");
    12. domains.put("org", "非营利组织");
    13. domains.put("edu", "教研机构");
    14. domains.put("gov", "政府部门");
    15. // 通过键获取值
    16. System.out.println("edu国际域名对应的含义为:" + domains.get("edu"));
    17. // 判断是否包含某个键
    18. System.out.println("domains键值对集合中是否包含gov:" + domains.containsKey("gov"));
    19. // 删除键值对
    20. domains.remove("gov");
    21. System.out.println("删除后集合中是否包含gov:" + domains.containsKey("gov"));
    22. // 输出全部键值对
    23. System.out.println(domains);
    24. }
    25. }

    1.6.2 Map 映射数据的遍历

            我们已经掌握了遍历 Collection 的通用方法—使用增强 for 或者迭代器 Iterator,并且知道 Collection 是单值形式元素的集合,而 Map 是键值对形式的元素集合。

            因此就能推测出,遍历 Map 的方法就是:先将 Map 集合或 Map 集合的部分元素转换成单值集合的形式,然后使用增强 for 或者迭代器 Iterator 遍历即可。

            简单来说就是:Map --> 转换为单值集合 --> 使用增强 for 或者迭代器 Iterator 遍历。

            我们可以将 Map 中的 key 全部提取出来,遍历 key,然后再根据 key 获取 value,如以下程序所示。

    1. import java.util.HashMap;
    2. import java.util.Iterator;
    3. import java.util.Map;
    4. import java.util.Set;
    5. /**
    6. * 采用 Iterator 方式遍历 Map 数据
    7. */
    8. public class TestMap {
    9. public static void main(String[] args) {
    10. Map map = new HashMap<>();
    11. map.put("k1", "v1");
    12. map.put("k2", "v2");
    13. map.put("k3", "v3");
    14. //将Map中的key全部提取出来
    15. Set keys = map.keySet();
    16. System.out.println("使用迭代器遍历");
    17. Iterator keyIter = keys.iterator();
    18. while (keyIter.hasNext()) {
    19. //获取map的每个key
    20. String key = keyIter.next();
    21. //根据key获取对应的value
    22. String value = map.get(key);
    23. System.out.println(key + "," + value);
    24. }
    25. System.out.println("使用增强for遍历");
    26. for (String key : keys) {
    27. //根据key获取对应的value
    28. String value = map.get(key);
    29. System.out.println(key + "," + value);
    30. }
    31. }
    32. }

            另一种方式遍历:Map 中的每一组 key-value 对称为一个 entry 对象,即 entry = key + value。Map 接口提供了获取 Map 中全部 entry 对象的方法,因此就可以先获取全部的 entry 对象,然后再提取 entry 对象中的 key 和 value。

    1. import java.util.HashMap;
    2. import java.util.Map;
    3. import java.util.Set;
    4. /**
    5. * 采用 Map.Entry 方式遍历 Map 数据
    6. */
    7. public class TestMap2 {
    8. public static void main(String[] args) {
    9. Map map = new HashMap<>();
    10. map.put("k1", "v1");
    11. map.put("k2", "v2");
    12. map.put("k3", "v3");
    13. //获取 Map 的全部 entry 对象
    14. Set> entries = map.entrySet();
    15. //遍历 entry 集合
    16. for (Map.Entry entry : entries) {
    17. String key = entry.getKey();
    18. String value = entry.getValue();
    19. System.out.println(key + "," + value);
    20. }
    21. }
    22. }

            控制台录入一个字符串,程序经过统计最后输出这个字符串中每个字母出现的次数。

    1. import java.util.*;
    2. public class Statistics {
    3. static HashMap hashMap = new HashMap<>();
    4. public Map statis(String str){
    5. // 判断 map里面是否存过key=c的键
    6. //有则 查找其value并且加1; 再存进map里面
    7. for (char c:str.toCharArray()){
    8. if(hashMap.containsKey(c)){
    9. int value=hashMap.get(c);
    10. value++;
    11. hashMap.put(c,value);
    12. }else {
    13. hashMap.put(c,1);
    14. }
    15. }
    16. return hashMap;
    17. }
    18. public static void main(String[] args) {
    19. Scanner input = new Scanner(System.in);
    20. System.out.println("请输入一个字符串:");
    21. String str=input.next();
    22. //将非字母的字符去掉
    23. String str1 = str.replaceAll("[^a-zA-Z]", "");
    24. Statistics statistics = new Statistics();
    25. statistics.statis(str1);
    26. Set set=hashMap.keySet();
    27. for (char c:set){
    28. int value=hashMap.get(c);
    29. System.out.println(c+"="+value);
    30. }
    31. }
    32. }

    1.7工具类

    1.7.1Collections 工具类的使用 

    Collections 工具类,是集合对象的工具类,类中方法都是静态的,可以直接以 类名.静态方法() 的形式调用。

    该类提供了操作集合的工具方法,如排序、复制、反转和查找等方法,如下所示:

    • void sort(List list)

      根据数据元素的排序规则对 List 集合进行排序,其中的排序规则是通过内部比较器设置的。例如 List 中存放的是 obj 对象,那么排序规则就是根据 obj 所属类重写内部比较器 Comparable 中的 compareTo() 方法定义的。

    • void sort(List list, Comparator c)

      根据指定比较器中的规则对 List 集合进行排序。通过自定义 Comparator 比较器 c,可以实现按程序员定义的规则进行排序。

    • void shuffle(List list)

      对指定 List 集合进行随机排序。

    • void reverse(List list)

      反转 List 集合中数据元素的顺序。

    • Object max(Collection coll)

      根据数据元素的自然顺序,返回给定 coll 集合中的最大元素。该方法的输入类型为 Collection 接口,而非 List 接口,因为求集合中最大元素不需要集合是有序的。

    • Object min(Collection coll)

      根据数据元素的自然顺序,返回给定 coll 集合的最小元素。

    • int binarySearch(List list,Object o)

      使用二分查找法查找 list 集合,以获得 o 数据元素的索引。如果此集合中不包含 o 元素,则返回-1。在进行此调用之前,必须根据 list 集合数据元素的自然顺序对集合进行升序排序(通过 sort(List list) 方法)。如果没有对 list 集合进行排序,则结果是不确定的。如果 list 集合中包含多个元素 “等于” 指定的 o 元素,则无法保证找到的是哪一个,这里说的 “等于” 是指通过 equals() 方法判断相等的元素。

    • int indexOfSubList(List source,List target)

      返回指定源集合 source 中第一次出现指定目标集合 target 的起始位置,换句话说,如果 target 是 source 的一个子集合,那么该方法返回 target 在 source 中第一次出现的位置。如果没有出现这种集合间的包含关系,则返回 -1。

    • int lastIndexOfSubList(List source,List target)

      返回指定源集合 source 中最后一次出现指定目标集合 target 的起始位置,如果没有出现这样的集合,则返回 -1。

    • void copy(List dest,List src)

      将所有数据元素从 src 集合复制到 dest 集合。

    • void fill(List list,Object o)

      使用 o 数据元素替换 list 集合中的所有数据元素。

    • boolean replaceAll(List list,Object old,Object new)

      使用一个指定的 new 元素替换 list 集合中出现的所有指定的 old 元素。

    • void swap(List list,int i,int j)

      在 list 集合中,交换 i 位置和 j 位置的元素。

    1. import java.util.ArrayList;
    2. import java.util.Collections;
    3. import java.util.List;
    4. public class TestCollections {
    5. public static void main(String[] args) {
    6. List list = new ArrayList();
    7. list.add("w");
    8. list.add("o");
    9. list.add("r");
    10. list.add("l");
    11. list.add("d");
    12. System.out.println("排序前: " + list);
    13. System.out.println("该集合中的最大值:" + Collections.max(list));
    14. System.out.println("该集合中的最小值:" + Collections.min(list));
    15. Collections.sort(list);
    16. System.out.println("sort排序后: " + list);
    17. //使用二分查找,查找前须保证被查找集合是自然有序排列的
    18. System.out.println("r在集合中的索引为: " + Collections.binarySearch(list, "r"));
    19. Collections.shuffle(list);
    20. System.out.println("再shuffle排序后: " + list);
    21. Collections.reverse(list);
    22. System.out.println("再reverse排序后: " + list);
    23. Collections.swap(list, 1, 4);
    24. System.out.println("索引为1、4的元素交换后:" + list);
    25. Collections.replaceAll(list, "w", "d");
    26. System.out.println("把w都换成d后的结果: " + list);
    27. Collections.fill(list, "s");
    28. System.out.println("全部填充为s后的结果: " + list);
    29. }
    30. }
    • 主要使用了 sort()、shuffle()、reverse()、max()、min()、binarySearch()、fill()、replaceAll()、swap() 方法。

    • 该工具类可以方便的完成对集合数据的一系列操作

    1.7.2 Arrays 工具类的使用

    Arrays 类是操作数组的工具类,和 Collections 工具类相似,Arrays 类主要有以下功能:

    • 对数组进行排序。
    • 给数组赋值。
    • 比较数组中元素的值是否相等。
    • 进行二分查找。
    1. import java.util.Arrays;
    2. /**
    3. * Arrays 工具类的使用
    4. */
    5. public class TestArrays {
    6. public static void output(int[] a) {
    7. for (int num : a) {
    8. System.out.print(num + " ");
    9. }
    10. System.out.println();
    11. }
    12. public static void main(String[] args) {
    13. int[] array = new int[5];
    14. //填充数组
    15. Arrays.fill(array, 8);
    16. System.out.println("填充数组Arrays.fill(array,8):");
    17. TestArrays.output(array);
    18. //将数组索引为1到4的元素赋值为6
    19. Arrays.fill(array, 1, 4, 6);
    20. System.out.println("将数组索引为1到4的元素赋值为6 Arrays.fill(array, 1, 4, 6):");
    21. TestArrays.output(array);
    22. int[] array1 = {12, 9, 21, 43, 15, 6, 19, 77, 18};
    23. //对数组索引为3到7的元素进行排序
    24. System.out.println("排序前,数组的序列为:");
    25. TestArrays.output(array1);
    26. Arrays.sort(array1, 3, 7);
    27. System.out.println("对数组索引为3到7的元素进行排序:Arrays.sort(array1,3,7):");
    28. TestArrays.output(array1);
    29. //对数组进行自然排序
    30. Arrays.sort(array1);
    31. System.out.println("对数组进行自然排序 Arrays.sort(array1):");
    32. TestArrays.output(array1);
    33. //比较数组元素是否相等
    34. int[] array2 = array1.clone();
    35. System.out.println("数组克隆后是否相等:Arrays.equals(array1, array2):" +
    36. Arrays.equals(array1, array2));
    37. //使用二分查找法查找元素下标(数组必须是排好序的)
    38. System.out.println("77在数组中的索引:Arrays.binarySearch(array1, 77):"
    39. + Arrays.binarySearch(array1, 77));
    40. }
    41. }

    综合案例

    某班有 40 个学生,学号为 180201-180240, 全部参加 Java 集合阶段检测,给出所有同学的成绩 (可随机产生,范围为 50-100),请编写程序将本班 各位同学成绩从高往低排序打印输出。

    注:成绩相同时学号较小的优先打印。

    1. package entity;
    2. public class Student {
    3. public int id;
    4. String name;
    5. public int score;
    6. @Override
    7. public String toString() {
    8. return "学号:" + id + ", 姓名:" + name + ", 成绩:" + score;
    9. }
    10. public Student() {
    11. }
    12. public Student(int id, String name, int score) {
    13. this.id = id;
    14. this.name = name;
    15. this.score = score;
    16. }
    17. public int getId() {
    18. return id;
    19. }
    20. public void setId(int id) {
    21. this.id = id;
    22. }
    23. public String getName() {
    24. return name;
    25. }
    26. public void setName(String name) {
    27. this.name = name;
    28. }
    29. public int getScore() {
    30. return score;
    31. }
    32. public void setScore(int score) {
    33. this.score = score;
    34. }
    35. }
    1. package main;
    2. import entity.Student;
    3. import java.util.*;
    4. public class Results {
    5. public static List data = new ArrayList();
    6. public void initData() {
    7. Random random = new Random();
    8. int k;
    9. for (int i = 0; i < 40; i++) {
    10. k = i + 1;
    11. Student student = new Student(180201 + i, ("J" + k), (random.nextInt(50) + 50));
    12. data.add(student);
    13. }
    14. }
    15. public void adjust() {
    16. Collections.sort(data, new GradeComparator());
    17. }
    18. public void print() {
    19. for (int i = 0; i < data.size(); i++) {
    20. Student s = data.get(i);
    21. System.out.println(s);
    22. }
    23. }
    24. //定义一个成绩比较器
    25. static class GradeComparator implements Comparator {
    26. @Override
    27. public int compare(Object o1, Object o2) {
    28. Student O1 = (Student) o1;
    29. Student O2 = (Student) o2;
    30. if(O2.score == O1.score){
    31. return O1.id-O2.id;
    32. }
    33. return O2.score - O1.score;
    34. }
    35. public static void main(String[] args) {
    36. Results results = new Results();
    37. results.initData();
    38. results.adjust();
    39. results.print();
    40. }
    41. }
    42. }

  • 相关阅读:
    王道数据结构(串4.1)
    PLC中ST编程的星角降压启动
    基于WiFi小车控制板的单片机小系统原理图
    并行与分布式计算 第三章 进程级的并行:MPI编程
    物联网安全概述
    隐私计算FATE-离线预测
    机器学习原理篇:基础数学理论 Ⅱ
    高并发下的网络 IO 模型设计
    基于单片机的太阳能热水器控制系统设计
    APP自动化测试-Appium Inspector入门操作指南
  • 原文地址:https://blog.csdn.net/qq_58631644/article/details/126042104