Interface 接口详解
简介
接口主要用来描述类具有哪些功能,并不给出每个功能的具体实现方式。一个类可以实现一个或多个接口,并在需要接口的地方,随时使用实现了响应接口的对象。
在 Java 程序设计语言中,接口不是类,而是对类的一组需求描述,这些类要遵循接口描述的统一格式进行定义。
接口中的所有方法自动属于 public,所以在接口中声明方法可以不用提供关键字 public 。
类实现接口,通常有两个步骤:
- 将类声明为实现指定的接口
- 对接口中的所有方法进行定义
要将类声明为实现某个接口,需要使用关键字 implements :
public interface Comparable{ int compareTo(Object other);}
class Employee implements Comparable{}
接口特性
接口不是类,尤其不能使用 new 运算符实例化一个接口。但是可以声明接口的变量,接口变量必须引用实现该接口的类对象。
X = new Comparable(...); //Error
Comparable x; // OK
x = new Employee(); // OK
与可以建立类的继承关系一样,接口也可以被扩展。这里允许存在多条从较高通用性的接口道较高专用型的忌口的链。
public interface Moveable{ void move(double x,double y);}
public interface Powerd extends Moveable{ double milesPerGallon();}
虽然在接口中不能包含实例域或静态方法,但可以包含常量。
public interface Powerd extends Moveable{
double milesPerGallon();
double SPEED_LIMIT = 95;
}
与接口中的方法都自动地被设置为 public 一样,接口中的域将自动设为 public static final 。
接口与抽象类
使用抽象类表示通用属性存在一个问题 :每个类只能扩展一个类。但是每个类可以实现多个接口。
静态方法
在 Java SE 8 中,允许在接口中增加静态方法。理论上,没有任何理由认为这是不合适的,只是这有违于将接口最为抽象规范的初衷。
public interface Path{
public static Path get(String first,String... more){
return FileSystems.getDefault().getPath(first,more);
}
...
}
默认方法
可以为接口方法提供一个默认实现。必须用 default 修饰符标记这样的方法。
public interface Comparable<T>{
default int compareTo(T other){
return 0; // 默认情况下,所有的数据都相等
}
}
这样可能没有什么用,因为 Comparable 的每一个实现都要覆盖这个方法。但是在某些情况下,默认方法可能很有用。例如:
public interface Collection{
int size();
default boolean isEmpty(){
return size() == 0;
}
...
}
这样在实现 Collection 的程序员就不用操心实现 isEmpty 方法了。
默认方法的一个重要用法是 “接口演化”。例, Collection 接口作为 Java 的一部分已经很多年了,在 Java SE 8中,为这个接口添加了一个 stream 方法。假设 stream 不是默认方法,那么引用 Collection 接口的类将不能编译,因为没有实现这个新方法。为接口增加一个非默认方法不能保证 “源代码兼容”。
不过,假设不重新编译这个类,而只是使用原先的一个包含这个类的 JAR 文件。这个类仍可以正常加载,尽管没有这个新方法。但是,如果程序在一个该类的实例上调用 stream 方法,就会出现一个 AbstractMethodError。
默认方法可以解决这些问题。如果没有重新编译而直接加载这个类,并在一个实例上调用 stream 方法,将调用 Collection stream 方法。
解决默认方法冲突
如果现在一个接口中将一个方法定义为默认方法,然后又在超类或者另一个接口中定义了同样的方法。会发生什么?Java对应的规则比较简单:
- 超类优先。如果超类提供了一个具体方法,接口中同名并且有相同参数类型的默认方法会被忽略。
- 接口冲突。如果一个超接口提供了一个某人方法,另一个接口提供了一个同名而且参数类型相同的方法,必须覆盖这个方法来解决冲突。
public interface Named{
default String getName(){
return getClass().getName()+"_"+hashCode();
}
}
Class Student implements Person,Named{
...
}
类会继承 Person 和 Named 接口提供的两个不一致的 getName 方法。并不是从中选择一个,Java 编译器会报告一个错误,让程序员来解决这个二义性。只需要在 Student 类中提供一个 getName 方法,在这个方法中,可以选择两个冲突方法中的一个。
Class Student implements Person,Named{
public String getName(){ return Person.super.getName();}
...
}
接口示例
接口与回调
回调是一种常见的程序设计模式。在这种模式中,可以指出某个特定事件发生时应该采取的行动。
例如:程序中有一个时钟,每隔十秒钟报一次时。
import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.util.Date;
import javax.swing.*;
public class TimerTest {
public static void main(String[] args) {
ActionListener listener = new TimePrinter();
Timer t = new Timer(10000, listener);
t.start();
JOptionPane.showMessageDialog(null, "Quit program?");
System.exit(0);
}
}
class TimePrinter implements ActionListener {
@Override
public void actionPerformed(ActionEvent e) {
System.out.println("At the tone, the time is " + new Date());
Toolkit.getDefaultToolkit().beep();
}
}
对象克隆
Cloneable 接口指示一个类提供了一个安全的 clone 方法。clone 产生的对象是一个新对象,它的初始状态与源对象相同,但是之后他们各自会有自己不同的状态。
不过, clone 方法是 Object 的一个 protected 方法,这说明你的代码不能直接调用这个方法。只有 Employee 类可以克隆 Employee 对象。
默认的克隆操作是“浅拷贝”,并没有克隆对象中引用的的其他对象。如果源对象和浅克隆对象共享的子对象是不可变的,这种共享就是安全的。如果子对象属于一个不可变的类,如 String ,这种共享是安全的。或者在对象的声明周期中,子对象一只包含不变的常量,没有更改器方法会改变它,也没有方法会生成它的引用,这种情况下同样是安全的。

不过,通常子对象都是可变的,必须重新定义 clone 方法来建立一个深拷贝,同时克隆所有子对象。
对于每一个类,需要确定:
- 默认的 clone 方法是否满足要求
- 是否可以在可变的子对象上调用 clone 来修补默认的 clone 方法
- 是否不改使用 clone
实际上第 3 个选项是默认选项。如果选择第 1 项或第 2 项,类必须:
- 实现 Cloneable 接口
- 重新定义 clone 方法,并指定 public 访问修饰符
Tips:
Object 类中的 clone 方法声明为 protected,子类只能调用受保护的 clone 方法来克隆自己的对象。必须重新定义 clone 为 public 才能允许所有方法克隆对象。
Cloneable 接口没有指定 clone 方法,这个方法是从 Object 类继承的。这个接口指示作为一个标记,指示类设计者了解克隆过程。标记接口不包含任何方法,它的唯一作用就是允许在类型查询中使用 instanceof。
即使 clone 的默认(浅拷贝)实现能满足要求,还是需要实现 Cloneable 接口,将 clone 重新定义为 public,再调用 super.clone()。
class Employee implements Cloneable{
public Employee clone() throws CloneNotSupportedException{
return (Employee) super.clone();
}
...
}
与 Object.clone 提供的浅拷贝相比,这个 clone 方法并没有为它添加任何工嗯呢该。只是让这个方法是公有的。要建立深拷贝,还需要做很多工作,克隆对象中可变的实例域。
class Employee implements Cloneable{
public Employee clone() throws CloneNotSupportedException{
Employee cloned = (Employee) super.clone();
cloned.hireDay = (Date) hireDay.clone();
return cloned;
}
...
}
如果在一个对象上调用 clone,但这个对象的类并没有实现 Cloneable 接口,Object 类的 clone 方法就会抛出一个 CloneNotSupportedException。
捕捉这个异常是不是更好一些?这非常适合 final 类。否则,还是保留 throws 说明符。这样就允许子类在不支持克隆时选择抛出一个 CloneNotSupportedException。
要不要在自己的类中实现 clone 呢?如果你的客户端需要建立深拷贝,可能就需要实现这个方法。
Tips:
所有数组类型都有一个 public 的 clone 方法,而不是 protected。可以用这个方法建立一个新数据,包含原数组所有元素的副本。
Interface 接口详解的更多相关文章
- Java接口 详解(二)
上一篇Java接口 详解(一)讲到了接口的基本概念.接口的使用和接口的实际应用(标准定义).我们接着来讲. 一.接口的应用—工厂设计模式(Factory) 我们先看一个范例: package com. ...
- [转载]MII/MDIO接口详解
原文地址:MII/MDIO接口详解作者:心田麦浪 本文主要分析MII/RMII/SMII,以及GMII/RGMII/SGMII接口的信号定义,及相关知识,同时本文也对RJ-45接口进行了总结,分析了在 ...
- JDBC常用接口详解
JDBC中常用接口详解 ***DriverManager 第一.注册驱动 第一种方式:DriverManager.registerDriver(new com.mysql.jdbc.Driver()) ...
- Java6.0中Comparable接口与Comparator接口详解
Java6.0中Comparable接口与Comparator接口详解 说到现在,读者应该对Comparable接口有了大概的了解,但是为什么又要有一个Comparator接口呢?难道Java的开发者 ...
- socket接口详解
1. socket概述 socket是在应用层和传输层之间的一个抽象层,它把TCP/IP层复杂的操作抽象为几个简单的接口供应用层调用已实现进程在网络中通信. socket起源于UNIX,在Unix一切 ...
- “全栈2019”Java第八十四章:接口中嵌套接口详解
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- “全栈2019”Java第八十三章:内部类与接口详解
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- map接口详解
1.Map接口详解(1)映射(map)是一个存储键.键值对的对象,给定一个键,可以查询得到它的值,键和值都可以是对象(2)键必须是唯一的,值可以重复(Map接口映射唯一的键到值)(3)有些映射可以接收 ...
- ReadWriteLock 接口详解
ReadWriteLock 接口详解 这是本人阅读ReadWriteLock接口源码的注释后,写出的一篇知识分享博客 读写锁的成分是什么? 读锁 Lock readLock(); 只要没有写锁,读锁可 ...
随机推荐
- UltraEdit文字编辑器菜单热键推荐
键盘映射和自定义菜单热键 任何使用过UltraEdit / UEStudio一段时间的人都可能会告诉您,他们如此喜欢它的原因之一是"几乎所有东西都是可定制的".看一下产品鉴定,您会 ...
- 再解决不了前端加密我就吃shi
参考文章 快速定位前端加密方法 渗透测试-前端加密测试 前言 最近学习挖洞以来,碰到数据做了加密基本上也就放弃了.但是发现越来越多的网站都开始做前端加密了,不论是金融行业还是其他.所以趁此机会来捣鼓一 ...
- macOS使用ABP.vNext Core开发CMS系统(一) 让程序跑起来
macOS使用ABP.vNext Core开发CMS系统(一) 让程序跑起来--2020年10月5日 国庆假期,陪老婆的同时也不能忘记给自己充充电,这不想搞个CMS系统,考虑自己的时间并不多,所以想找 ...
- Win32控制台、Win32项目、MFC项目、CLR控制台、CLR空项目、空项目区别
转载:https://blog.csdn.net/zfmss/article/details/79244696 1.Win32控制台 初始代码模版以main为程序入口,默认情况下,只链接C++运行时库 ...
- Tomcat 第六篇:类加载机制
1. 引言 Tomcat 在部署 Web 应用的时候,是将应用放在 webapps 文件夹目录下,而 webapps 对应到 Tomcat 中是容器 Host ,里面的文件夹则是对应到 Context ...
- 《流畅的Python》第三部分 把函数视作对象 【一等函数】【使用一等函数实现设计模式】【函数装饰器和闭包】
第三部分 第5章 一等函数 一等对象 在运行时创建 能赋值给变量或数据结构中的元素 能作为参数传递给函数 能作为函数的返回结果 在Python中,所有函数都是一等对象 函数是对象 函数本身是 func ...
- ERROR 1045 (28000): Access denied for user 'root'@'localhost' (using password: YES)解决方案
在Win7下使用MySQL5.6.35创建用户时,提示权限不足,具体解决方案如下: 1 停止mysql服务 net stop mysql 2 打开新的cmd窗口,切换到bin目录,运行如下命令,cmd ...
- 多测师讲解自动化测试 _如何解决验证码的问题_高级讲师肖sir
自动化测试如何解决验证码的问题对于web应用来说,大部分的系统在用户登录时都要求用户输入验证码,验证码的类型的很多,有字母数字的,有汉字的,甚至还要用户输入一条算术题的答案的,对于系统来说使用验证码可 ...
- 阿里百秀后台管理项目笔记 ---- Day01
摘要 在此记录一下阿里百秀项目的教学视频的学习笔记,部分页面被我修改了,某些页面效果会不一样,基本操作是一致的,好记性不如烂笔头,加油叭!!! step 1 : 整合全部静态页面 将静态页面全部拷贝到 ...
- MeteoInfoLab脚本示例:Streamline流线图
绘制Stramline流线图的函数是streamline,需要两个变量(U/V分量或者风向/风速).脚本程序: f = addfile('D:/Temp/GrADS/model.ctl') u = f ...