接口使用实例
给对象数组排序
两个学生对象的大小关系怎么确定? 需要我们额外指定.
这里需要用到Comparable 接口
在Comparable 接口内部有一个compareTo 的方法,我们需要实现它
在下图中,我们需要将o强制转换为Student
之后调用Arrays.sort(array)即可完成排序,如下
代码:
package demo5;import java.util.Arrays;/*** Created with IntelliJ IDEA.* Description:* User: Home-pc* Date: 2023-08-16* Time: 16:40*/
class Student implements Comparable{public String name;public double score;public Student(String name,double score){this.name=name;this.score=score;}@Overridepublic String toString() {return "Student{" +"name='" + name + '\'' +", score='" + score + '\'' +'}';}public int compareTo(Object o) {Student student=(Student)o;if(this.score>student.score){return 1;}else if(this.score<student.score){return -1;}return 0;}
}
public class Test {public static void main(String[] args) {Student[] array=new Student[3];array[0]=new Student("zhangsan",80);array[1]=new Student("qwer",7);array[2]=new Student("asdfg",806);System.out.println("前"+ Arrays.toString(array));Arrays.sort(array);System.out.println("后"+ Arrays.toString(array));}
}
为了进一步加深对接口的理解, 我们可以尝试自己实现一个 sort 方法来完成刚才的排序过程(使用冒泡排序)
结果也是正确的
扩展上面的内容,我们想分别按照年龄和分数排序
需要用到Comparator接口
建立两个不同的类来实现这两种方法,都要实现Comparator接口
结果如下
上述代码如下:
package demo5;import java.util.Arrays;
import java.util.Comparator;/*** Created with IntelliJ IDEA.* Description:* User: Home-pc* Date: 2023-08-16* Time: 16:40*/
class Student{public String name;public double score;public int age;public Student(String name,double score,int age){this.name=name;this.score=score;this.age=age;}@Overridepublic String toString() {return "Student{" +"name='" + name + '\'' +", score='" + score + '\'' +'}';}
}
class AgeComparator implements Comparator<Student>{@Overridepublic int compare(Student o1, Student o2) {return o1.age-o2.age;}
}
class ScoreComparator implements Comparator<Student>{@Overridepublic int compare(Student o1, Student o2) {return (int)(o1.score-o2.score);}
}
public class Test {public static void main(String[] args) {Student[] array=new Student[3];array[0]=new Student("zhangsan",80,89);array[1]=new Student("qwer",7,123);AgeComparator ageComparator=new AgeComparator();System.out.println(ageComparator.compare(array[0], array[1]));System.out.println("@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@");ScoreComparator scoreComparator=new ScoreComparator();System.out.println(scoreComparator.compare(array[0], array[1]));}
}
抽象类和接口的区别
核心区别: 抽象类中可以包含普通方法和普通字段, 这样的普通方法和字段可以被子类直接使用(不必重写), 而接口中
不能包含普通方法, 子类必须重写所有的抽象方法.
抽象类存在的意义是为了让编译器更好的校验, 像 Animal 这样的类我们并不会直接使用, 而是使用它的子类.
万一不小心创建了 Animal 的实例, 编译器会及时提醒我们.
Clonable 接口和深浅拷贝
Object 类中存在一个 clone 方法, 调用这个方法可以创建一个对象的 “拷贝”. 但是要想合法调用 clone 方法, 必须要
先实现 Clonable 接口
如下所示,我们需要Student实现这个接口,需要对clone方法重写
但是我们运行后发现还是报错
我们发现这里,点击添加Add
运行发现继续报错,观察到红线这里,需要的类型为Student,我们提供的类型为Object,需要强制类型转换
运行,拷贝成功
上述为浅拷贝
结果并不理想,它把两个对象里的grade都改了,这种情况下没有构成深拷贝,拷贝的是grade的地址,它只是对Student对象中的内容进行了一份拷贝,但是并没有对,对象中的对象中的内容进行拷贝。
如何进行深拷贝?
我们改动以下部分
Object类
Object是Java默认提供的一个类。Java里面除了Object类,所有的类都是存在继承关系的。默认会继承Object父
类。即所有类的对象都可以使用Object的引用进行接收。
使用Object接收所有类的对象
Object类是参数的最高统一类型。但是Object类也存在有定义好的一些方法
我们主要来熟悉这几个方法:toString()方法,equals()方法,hashcode()方法
Object类中的toString()方法实现,可以直接用编译器生成重写
对象比较equals方法
在Java中, = = == ==进行比较时:
a.如果 = = == ==左右两侧是基本类型变量,比较的是变量中值是否相同
b.如果 = = == ==左右两侧是引用类型变量,比较的是引用变量地址是否相同
c.如果要比较对象中内容,必须重写Object中的equals方法,因为equals方法默认也是按照地址比较的
结论:比较对象中内容是否相同的时候,一定要重写equals方法
hashcode方法
算出一个具体的对象位置
我们认为两个名字相同,年龄相同的对象,将存储在同一个位置,如果不重写hashcode()方法,我们可以来看示例
代码:
像重写equals方法一样,我们也可以重写hashcode()方法。