前言
在Java中,我们使用接口优先原则,当一个场景可以使用抽象类也可以使用接口定义时,优先考虑使用接口,因为接口更灵活,使用抽象类必须满足is a 的继承树关系,而且是单继承,接口相比于抽象类更加的灵活,本篇我们就一起看看JDK中的常用内置接口。
一、comparable接口
java.lang.Comperable:
当一个类实现了Comperable接口,就表示该类具备了可比较的能力!
此时定义了一个Person类型的数组:
由于Person这个类型是自定义的类型,对于编译器来说,不像int类型一样大小关系一目了然;
对于编译器来说,到底哪个Person对象大,哪个Person对象小,编译器就无从得知了,此时就要让Person这个类型具备可比较的能力,也就是要让JDK知道Person对象“谁大谁小”,就需要让Person这个类实现Comperable接口,覆写抽象方法compereTo。
此时覆写的方法返回值是int,传入的参数是任意类型,因为不知道要比较的是什么类型
int返回值 >0 =0 <0 表示当前对象this > 传入对象o 表示当前对象this = 传入对象o 表示当前对象this < 传入对象o
覆写comperTo方法:
当Person类实现了Comperable接口,覆写了compereTo方法之后,就可以把Person 类的对象进行大小排序了,可以按年龄大小进行比较,也可以按照姓名大小比较
此时是默认按照年龄的升序排序的,如果想要按照年龄的降序排序,该怎么办?该如何调整compereTo方法呢?
此时只需要将返回值取反,让传入的年龄减去当前对象的年龄,让JDK误认为年龄越小的反而越“大”。
二、Cloneable接口
java.lang.Cloneable
首先了解一下什么是克隆,就是原对象和新产生对象是两个独立的对象,新产生的对象是通过原对象“拷贝”而来的,属性值和原对象完全一致。
要想让一个类具备可复制的能力,就让该类实现Cloneable接口,覆写clone方法
这里Animal实现了Cloneable接口,但是没有覆写任何方法,但是这里没有报错,这是因为Cloneable接口里本来什么都没有,但是这并不是一个空接口
类似Cloneable接口,把这种接口称之为“标记”接口,这个接口本身内部没有任何抽象方法,只有打上这个“标记”的子类才具备可克隆的能力
JVM在运行时会检查所有实现了Cloneable接口的子类,然后赋予其可以克隆的能力
就像现在的一码通,根据一码通的颜色,把不同的人划分为不同的类,只有检测为绿码的人,才能通过。
animal1 是通过 animal 克隆来的,但是此时使用“ == ”比较他们两个的地址返回的还是false,说明克隆是产生了新的对象,只是新对象和原来的对象的属性值完全相同
此时在产生一个对象animal2
Animal animal2 = new Animal();
animal1和animal2的区别:
animal1是依赖于animal产生的;
animal2的产生和animal完全没有任何关系。
调用clone方法产生的对象不会调用构造方法:
在Java中产生对象一共有两种方式:
- 1.最普遍的通过构造方法产生对象,当有new关键字,就在堆上开辟该类相应属性的空间,给属性赋默认值。
- 2.通过clone()产生对象,调用clone方法时,JVM会开辟与原对象内存大小完全相同的新空间,并将对象的属性值从原对象中复制一份(不推荐,Java都不推荐使用这种方式产生对象)
三、深浅拷贝
1.浅拷贝
克隆对象和原对象的确是两个独立的对象,
但是b1和b2的对象内部包含了其他的引用,克隆后的b2包含的其他引用并没有产生新的对象,b1.a 和 b2.a指向相同的A对象。
2.深拷贝
深拷贝就是克隆对象内部包含的其他引用,也产生了新的对象
b1.a 和 b2.a指向不同的A对象。
Java中深拷贝的实现方式:
- 1.递归的使用clone方法
- 2.序列化(json字符串)
到此这篇关于Java JDK内置常用接口和深浅拷贝的文章就介绍到这了,更多相关Java JDK内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!