Java 核心技术:接口详解
接口的概念
写实概念:接口不是类,而是对类的一组需求的描述。(可以看做是没有实例域的抽象类)
抽象概念:
接口的特性
- 接口中的方法自动地属于public,因此在接口中声明放方法是不必提供关键字public。
- 这么做的理由其实很好理解,接口本就是对类需求的一组描述,所以单独存在就没有意义,需要被类实现,所以其中方法的方法权限应当为public。
接口中绝对不能有实例域,可以定义常量,有些接口只定义了常量而没有方法(似乎偏离了接口的概念)。
java SE 8之前,不能在接口中实现方法,现在可以提供简单的实现。
类实现一个接口,应当对接口中的所有方法进行定义。
接口不是类,接口变量不能用new运算符实例化一个接口 。例如:
Comparable x = new Comparable(...);
接口变量可以引用实现了该接口的类对象。例如:Comparable x = new Employee(....)// Emploee provided implements Comparable
接口也可以被扩展。
public interface Moveable
{
void move(double x,double y);
}
public interface Powered extends Moveable
{
double milesPerGallon();
double SPEED_LIMIT = 95;//a public static final constant
}
一个类可以继承多个类。
class Employee implements a,b;
可以为接口中的方法提供一个默认实现,但必须用default修饰符标记该方法。
当类实现一个接口必须定义该接口中的所有方法,这又扯到单一职责原则。提供默认实现是用户只关注与那些需要关注的方法。
当A接口中只有一个方法a,突然A接口要增加b方法,这个时候如果b提供默认实现,就可以让使用之前版本的A接口的项目不会出错,同时又可以让该项目在拓展时,使用b方法。
接口和抽象类
-
既然说接口相似于没有实例域的抽象类,那为什么需要接口
1.一个类可以实现多个接口,但是只能继承一个类
解决默认方法的冲突
超类优先:顾名思义,只要超类有与接口一样的方法。
接口冲突:若其中有一个方法有默认实现,报告错误,让程序员解决该二义性。若都没有实现,可以选择实现这个方法,也可以选择不实现(此时这个类本身就是抽象类)
接口实例
接口与回调
- 回调是一种常见的设计模式。在这种模式中,可以指定某个特定事件发生时应该采取的动作。
- 定时器,他将某个类的对象传递给定时器,然后定时器调用这个对象的方法。该对象所属的类需要实现ActionListener接口,实现接口方法actionPerdformed()。定时器就知道执行哪个方法。(此处应当有图,暂时没有了)
对象克隆
- 第一个图是,两个对象变量的引用,要想达到二三图的克隆可以使用object类中的protect方法clone(),直接上代码。
//一图
Employee original = new Employee("wang",1000);
Employee copy = original;
//二图
Employee original = new Employee("wang",1000);
Employee copy = original.clone;
- 以上只是浅拷贝,如果在对象中有子对象,并且子对象还是可变的,此时可能需要深拷贝。我们需要重新定义clone()方法,并指定public访问修饰符。
- 另外在实现克隆过程中我们需要实现Cloneable接口,它没有方法,所以不用重定义,只是作为一个克隆标记。
public Employee implements Cloneable{
...
public Employee clone(){
Employee clone = (Employee)super.clone;
clone.hireDay = (Date)hireDay.clone;
}
}
- 若Employee的子类Manager也需要克隆,该方法可以完成吗
clone在平时项目的开发中可能用的不是很频繁,但是区分深拷贝和浅拷贝会让我们对java内存结构和运行方式有更深的了解。至于彻底深拷贝,几乎是不可能实现的,原因已经在上一节中进行了说明。深拷贝和彻底深拷贝,在创建不可变对象时,可能对程序有着微妙的影响,可能会决定我们创建的不可变对象是不是真的不可变。clone的一个重要的应用也是用于不可变对象的创建。关于创建不可变对象,我会在后续的文章中进行阐述,敬请期待。
protected 为保护类型,在同一个包里和 public 类型是一样的,也是能够访问到的。但是如果在不同包里的 protected 类型的成员变量就只能通过子类来访问,这个修饰符是区别于其他的修饰符的