[ Java学习基础 ] Java的抽象类与接口
一、抽象类
1. 抽象类
Java语言提供了两种类:一种是具体类;另一种是抽象子类。
2. 抽象类概念:
在面向对象的概念中,所有的对象都是通过类来描绘的,但是反过来,并不是所有的类都是用来描绘对象的,如果一个类中没有包含足够的信息来描绘一个具体的对象,这样的类就是抽象类。抽象类除了不能实例化对象之外,类的其它功能依然存在,成员变量、成员方法和构造方法的访问方式和普通类一样。
由于抽象类不能实例化对象,所以抽象类必须被继承,才能被使用。也是因为这个原因,通常在设计阶段决定要不要设计抽象类。父类包含了子类集合的常见的方法,但是由于父类本身是抽象的,所以不能使用这些方法。在Java中抽象类表示的是一种继承关系,一个类只能继承一个抽象类,而一个类却可以实现多个接口。
在[ Java学习基础 ] Java的继承与多态中介绍多态时,使用过几何图形类示例,其中Figure(几何图形)类中有一个onDraw(绘图)方法,Figure有两个子类Ellipse(椭圆形)和Triangle(三角形),Ellipse和Triangle覆盖onDraw方法。
作为父类Figure(几何图形)并不知道在实际使用时有多少个子类,目前有椭圆形和三角形,那么不同的用户需求可能会有矩形或圆形等其他几何图形,而onDraw方法只有确定是哪一个子类后才能具体实现。Figure中的onDraw方法不能具体实现,所以只能是一个抽象方法。在Java中具有抽象方法的类称为“抽象类”,Figure是抽象类,其中的onDraw方法是抽象方法。如下图所示类图中Figure是抽象类,Ellipse和Triangle是Figure子类实现Figure的抽象方法onDraw。
Tips:在UML类图抽象类和抽象方法字体是斜体的,如上图所示中的Figure类和onDraw方法都是斜体的。
3. 抽象类声明和实现
设计抽象方法目的就是让子类来实现的,否则抽象方法就没有任何意义,实现抽象类示例代码如下:
//Ellipse.java文件
package com.Kevin; //几何图形椭圆形
public class Ellipse extends Figure { //绘制几何图形方法
@Override
public void onDraw() {
System.out.println("绘制椭圆形...");
}
} //Triangle.java文件
package com.Kevin; //几何图形三角形
public class Triangle extends Figure { // 绘制几何图形方法
@Override
public void onDraw() {
System.out.println("绘制三角形...");
}
}
上述代码声明了两个具体类Ellipse和Triangle,它们实现(覆盖)了抽象类Figure的抽象方法onDraw。
调用代码如下:
//HelloWorld.java文件
package com.Kevin; public class HelloWorld { public static void main(String[] args) { // f1变量是父类类型,指向子类实例,发生多态
Figure f1 = new Triangle();
f1.onDraw(); // f2变量是父类类型,指向子类实例,发生多态
Figure f2 = new Ellipse();
f2.onDraw();
}
}
上述代码中实例化两个具体类Triangle和Ellipse,对象f1和f2是Figure引用类型。
声明抽象方法会造成以下两个结果:
- 如果一个类包含抽象方法,那么该类必须是抽象类。
- 任何子类必须重写父类的抽象方法,或者声明自身为抽象类。
Tips: 抽象类不能被实例化,只有具体类才能被实例化。
4. 抽象类总结规定:
- 抽象类不能被实例化(初学者很容易犯的错),如果被实例化,就会报错,编译无法通过。只有抽象类的非抽象子类可以创建对象。
- 抽象类中不一定包含抽象方法,但是有抽象方法的类必定是抽象类。
- 抽象类中的抽象方法只是声明,不包含方法体,就是不给出方法的具体实现也就是方法的具体功能。
- 构造方法,类方法(用static修饰的方法)不能声明为抽象方法。
- 抽象类的子类必须给出抽象类中的抽象方法的具体实现,除非该子类也是抽象类。
二、使用接口
接口(英文:Interface),在JAVA编程语言中是一个抽象类型,是抽象方法的集合,接口通常以interface来声明。一个类通过继承接口的方式,从而来继承接口的抽象方法。
接口并不是类,编写接口的方式和类很相似,但是它们属于不同的概念。类描述对象的属性和方法。接口则包含类要实现的方法。
除非实现接口的类是抽象类,否则该类要定义接口中的所有方法。
接口无法被实例化,但是可以被实现。一个实现接口的类,必须实现接口内所描述的所有方法,否则就必须声明为抽象类。另外,在 Java 中,接口类型可用来声明一个变量,他们可以成为一个空指针,或是被绑定在一个以此接口实现的对象。
1.接口声明和实现
在Java中接口的声明使用的关键字是interface,声明接口Figure示例代码如下:
//Figure.java文件
package com.Kevin; public interface Figure {
//接口中静态成员变量
String name = "几何图形";//省略public static final // 绘制几何图形方法
void onDraw(); //省略public
}
代码第4行是声明Figure接口,声明接口使用interface关键字,interface前面的修饰符是public或省略。public是公有访问级别,可以在任何地方访问。省略是默认访问级别,只能在当前包中访问。
代码第6行声明接口中的成员变量,在接口中成员变量都静态成员变量,即省略了public static final修饰符。代码第99行是声明抽象方法,即省略了public关键字。
某个类实现接口时,要在声明时使用implements关键字,当实现多个接口之间用逗号(,)分隔。实现接口时要实现接口中声明的所有方法。
实现接口Figure示例代码如下:
//Ellipse.java文件
package com.Kevin.imp; import com.Kevin.Figure; //几何图形椭圆形
public class Ellipse implements Figure { //绘制几何图形方法
@Override
public void onDraw() {
System.out.println("绘制椭圆形...");
}
} //Triangle.java文件
package com.Kevin.imp; import com.Kevin.Figure; //几何图形三角形
public class Triangle implements Figure { // 绘制几何图形方法
@Override
public void onDraw() {
System.out.println("绘制三角形...");
}
}
上述代码声明了两个具体类Ellipse和Triangle,它们实现了接口Figure中的抽象方法onDraw。
调用代码如下:
//HelloWorld.java文件
import com.Kevin.imp.Ellipse;
import com.Kevin.imp.Triangle; public class HelloWorld { public static void main(String[] args) { // f1变量是父类类型,指向子类实例,发生多态
Figure f1 = new Triangle();
f1.onDraw();
System.out.println(f1.name);
System.out.println(Figure.name); // f2变量是父类类型,指向子类实例,发生多态
Figure f2 = new Ellipse();
f2.onDraw();
}
}
上述代码中实例化两个具体类Triangle和Ellipse,对象f1和f2是Figure接口引用类型。接口Figure中声明了成员变量,它是静态成员变量,代码第12行和第13行是访问name静态变量。
Tips: 接口与抽象类一样都不能被实例化。
2.接口与多继承
在C++语言中一个类可以继承多个父类,但这会有潜在的风险,如果两个父类在有相同的方法,那么子类将继承哪一个父类方法呢?这就是C++多继承所导致的冲突问题。
在Java中只允许继承一个类,但可实现多个接口。通过实现多个接口方式满足多继承的设计需求。如果多个接口中即便有相同方法,它们也都是抽象的,子类实现它们不会有冲突。如下图所示是多继承类图,其中的有两个接口InterfaceA和InterfaceB,从类图中可以见两个接口中都有一个相同的方法void methodB()。AB实现了这两个接口,继承了Object父类。
接口InterfaceA和InterfaceB代码如下:
//InterfaceA.java文件
package com.Kevin; public interface InterfaceA { void methodA(); void methodB();
} //InterfaceB.java文件
package com.Kevin; public interface InterfaceB { void methodB(); void methodC();
}
从代码中可见两个接口都有两个方法,其中方法methodB()完全相同。
实现接口InterfaceA和InterfaceB的AB类代码如下:
//AB.java文件
package com.Kevin.imp; import com.Kevin.InterfaceA;
import com.Kevin.InterfaceB; public class AB extends Object implements InterfaceA, InterfaceB { @Override
public void methodC() {
} @Override
public void methodA() {
} @Override
public void methodB() {
}
}
在AB类中的代码第18行实现methodB()方法。注意在AB类声明时,实现两个接口,接口之间使用逗号(,)分隔,见代码第7行。
3.接口继承
Java语言中允许接口和接口之间继承。由于接口中的方法都是抽象方法,所以继承之后也不需要做什么,因此接口之间的继承要比类之间的继承简单的多。如下图所示,其中InterfaceB继承了InterfaceA,在InterfaceB中还覆盖了InterfaceA中的methodB()方法。ABC是InterfaceB接口的实现类,从下图中可见ABC需要实现InterfaceA和InterfaceB接口中的所有方法。
接口InterfaceA和InterfaceB代码如下:
//InterfaceA.java文件
package com.Kevin; public interface InterfaceA { void methodA(); void methodB();
} //InterfaceB.java文件
package com.Kevin; public interface InterfaceB extends InterfaceA { @Override
void methodB(); void methodC();
}
InterfaceB继承了InterfaceA,声明时也使用extends关键字。InterfaceB 中的methodB()覆盖了InterfaceA,事实上在接口中覆盖方法,并没有实际意义,因为它们都是抽象的,都是留给子类实现的。
实现接口InterfaceB的ABC类代码如下:
//ABC.java文件
package com.Kevin.imp; import com.Kevin.InterfaceB; public class ABC implements InterfaceB { @Override
public void methodA() {
} @Override
public void methodB() {
} @Override
public void methodC() {
}
}
ABC类实现了接口InterfaceB,事实上是实现InterfaceA和InterfaceB中所有方法,相当于同时实现InterfaceA和InterfaceB接口。
4.标记接口
最常用的继承接口是没有包含任何方法的接口。
标记接口是没有任何方法和属性的接口.它仅仅表明它的类属于一个特定的类型,供其他代码来测试允许做一些事情。
标记接口作用:简单形象的说就是给某个对象打个标(盖个戳),使对象拥有某个或某些特权。
例如:java.awt.event 包中的 MouseListener 接口继承的 java.util.EventListener 接口定义如下:
package java.util;
public interface EventListener
{}
没有任何方法的接口被称为标记接口。标记接口主要用于以下两种目的:
- 建立一个公共的父接口:
正如EventListener接口,这是由几十个其他接口扩展的Java API,你可以使用一个标记接口来建立一组接口的父接口。例如:当一个接口继承了EventListener接口,Java虚拟机(JVM)就知道该接口将要被用于一个事件的代理方案。
- 向一个类添加数据类型:
这种情况是标记接口最初的目的,实现标记接口的类不需要定义任何接口方法(因为标记接口根本就没有方法),但是该类通过多态性变成一个接口类型。
三、Java8新特性(默认方法和静态方法)
在Java 8之前,尽管Java语言中接口已经非常优秀了,但相比其他面向对象的语言而言Java接口存在如下不足之处:
不能可选实现方法,接口的方法全部是抽象的,实现接口时必须全部实现接口中方法,哪怕是有些方法并不需要,也必须实现。
没有静态方法。
针对这些问题,Java 8在接口中提供了声明默认方法和静态方法的能力。接口示例代码如下:
//InterfaceA.java文件
package com.Kevin; public interface InterfaceA { void methodA(); String methodB(); // 默认方法
default int methodC() {
return 0;
} // 默认方法
default String methodD() {
return "这是默认方法...";
} // 静态方法
static double methodE() {
return 0.0;
}
}
在接口InterfaceA中声明了两个抽象方法methodA和methodB,两个默认方法methodC和methodD,还有声明了静态方法methodE。接口中的默认方法类似于类中具体方法,给出了具体实现,只是方法修饰符是default。接口中静态方法类似于类中静态方法。
实现接口示例代码如下:
//ABC.java文件
package com.Kevin.imp; import com.Kevin.InterfaceA; public class ABC implements InterfaceA { @Override
public void methodA() {
} @Override
public String methodB() {
return "实现methodB方法...";
} @Override
public int methodC() {
return 500;
}
}
实现接口时接口中原有的抽象方法在实现类中必须实现。默认方法可以根据需要有选择实现(覆盖)。静态方法不需要实现,实现类中不能拥有接口中的静态方法。
上述代码中ABC类实现了InterfaceA接口,InterfaceA接口中的两个默认方法ABC只是实现(覆盖)了methodB。
调用代码如下:
//HelloWorld.java文件
package com.Kevin.imp; import com.Kevin.InterfaceA; public class HelloWorld { public static void main(String[] args) { //声明接口类型,对象是实现类,发生多态
InterfaceA abc = new ABC(); // 访问实现类methodB方法
System.out.println(abc.methodB()); // 访问默认方法methodC
System.out.println(abc.methodC()); // 访问默认方法methodD
System.out.println(abc.methodD()); // 访问InterfaceA静态方法methodE
System.out.println(InterfaceA.methodE());
}
}
运行结果:
实现methodB方法...
500
这是默认方法...
0.0
从运行结果可见,代码第17行调用默认方法methodC,是调用类AB中的实现。代码第20行调用默认方法methodD,是调用接口InterfaceA中的实现。代码第23行调用接口静态方法,只能通过接口名(InterfaceA)调用,不能通过实现类ABC调用,可以这样理解接口中声明的静态方法与其他实现类没有任何关系。
Tips:学习了接口默认方法后,有些读者还会有这样的疑问,Java 8之后接口可以声明抽象方法和具体方法,这就相当于抽象类一样了吗?在多数情况下接口不能替代抽象类,例如当需要维护一个对象的信息和状态时只能使用抽象类,而接口不行,因为维护一个对象的信息和状态需要存储在实例成员变量中,而接口中不能声明实例成员变量。
四、总结
1 接口与类的相似点:
- 一个接口可以有多个方法。
- 接口文件保存在 .java 结尾的文件中,文件名使用接口名。
- 接口的字节码文件保存在 .class 结尾的文件中。
- 接口相应的字节码文件必须在与包名称相匹配的目录结构中。
2 接口与类的区别:
- 接口不能用于实例化对象。
- 接口没有构造方法。
- 接口中所有的方法必须是抽象方法。
- 接口不能包含成员变量,除了 static 和 final 变量。
- 接口不是被类继承了,而是要被类实现。
- 接口支持多继承。
3 接口特性:
- 接口中每一个方法也是隐式抽象的,接口中的方法会被隐式的指定为 public abstract(只能是 public abstract,其他修饰符都会报错)。
- 接口中可以含有变量,但是接口中的变量会被隐式的指定为 public static final 变量(并且只能是 public,用 private 修饰会报编译错误)。
- 接口中的方法是不能在接口中实现的,只能由实现接口的类来实现接口中的方法。
4 接口与抽象类的区别:
- 抽象类中的方法可以有方法体,就是能实现方法的具体功能,但是接口中的方法不行。
- 抽象类中的成员变量可以是各种类型的,而接口中的成员变量只能是 public static final 类型的。
- 一个类只能继承一个抽象类,而一个类却可以实现多个接口。
5 什么时候使用抽象类和接口
- 如果你拥有一些方法并且想让它们中的一些有默认实现,那么使用抽象类吧。
- 如果你想实现多重继承,那么你必须使用接口。由于Java不支持多继承,子类不能够继承多个类,但可以实现多个接口。因此你就可以使用接口来解决它。
- 如果基本功能在不断改变,那么就需要使用抽象类。如果不断改变基本功能并且使用接口,那么就需要改变所有实现了该接口的类。
[ Java学习基础 ] Java的抽象类与接口的更多相关文章
- [ Java学习基础 ] Java的继承与多态
看到自己写的东西(4.22的随笔[ Java学习基础 ] Java构造函数)第一次达到阅读100+的成就还是挺欣慰的,感谢大家的支持!希望以后能继续和大家共同学习,共同努力,一起进步!共勉! ---- ...
- [ Java学习基础 ] Java构造函数
构造方法是类中特殊方法,用来初始化类的实例变量,它在创建对象(new运算符)之后自动调用. Java构造方法的特点如下: 构造方法名必须与类名相同. 构造方法没有任何返回值,包括void. 构造方法只 ...
- [ Java学习基础 ] Java的对象容器 -- 集合
当你有很多书时,你会考虑买一个书柜,将你的书分门别类摆放进入.使用了书柜不仅仅使房间变得整洁,也便于以后使用书时方便查找.在计算机中管理对象亦是如此,当获得多个对象后,也需要一个容器将它们管理起来,这 ...
- java面试基础题------》抽象类和接口有什么异同
划重点!!!! 1.抽象类(abstract class)和接口(interface)有什么异同? 相同点 * 都不能被直接实例化,都可以通过继承实现其抽象方法. * 都是面向抽象编程的技术基础,实现 ...
- java面向对象基础(四):抽象类和接口
*/ .hljs { display: block; overflow-x: auto; padding: 0.5em; color: #333; background: #f8f8f8; } .hl ...
- [ Java学习基础 ] Java异常处理
一.异常概述 异常是程序中的一些错误,但并不是所有的错误都是异常,并且错误有时候是可以避免的.比如说,你的代码少了一个分号,那么运行出来结果是提示是错误 java.lang.Error:如果你用Sys ...
- [ Java学习基础 ] Java对象的创建和销毁
类实例化可生成对象,实例方法就是对象方法,实例变量就是对象属性.一个对象的生命周期包括三个阶段:创建.使用和销毁. 创建对象 创建对象包括两个步骤:声明和实例化. 声明 声明对象与声明普通变量没有区别 ...
- [ Java学习基础 ] Java的封装性与访问控制
Java面向对象的封装性是通过对成员变量和方法进行访问控制实现的,访问控制分为4个等级:私有.默认.保护和公有,具体规则如下表: 1.私有级别 私有级别的关键字是private,私有级别的成员变量和方 ...
- Java学习--基础
java学习 基础 Java三大版本 javase 标准版 占领桌面端(基础) javame 移动版 嵌入式开发.占领手机端 javaee 企业版 占领服务器端 Java的特性和优势 跨平台.可移植性 ...
随机推荐
- 初学者如何查阅自然语言处理(NLP)领域学术资料
1. 国际学术组织.学术会议与学术论文 自然语言处理(natural language processing,NLP)在很大程度上与计算语言学(computational linguistics,CL ...
- (转载) ASP.NET(C#) Web Api 通过文件流下载文件到本地实例
下载文件到本地是很多项目开发中需要实现的一个很简单的功能.说简单,是从具体的代码实现上来说的,.NET的文件下载方式有很多种,本示例给大家介绍的是ASP.NET Web Api方式返回HttpResp ...
- zuul入门(5)zuul 处理异常
Object accessToken = request.getParameter("accessToken"); if(accessToken==null) { // 设置zuu ...
- 关于CheckStyle在eclipse出现的问题
今天在公司换了一个CheckStyle xml文件.那么我尝试直接import进去新的文件. 在我Check code的时候就爆了下面的错误 o: Failed during checkstyle c ...
- Oracle:常用的一些基本操作
表操作 查看系统中当前用户所有表: select * from user_tables;select * from user_indexs;select * from user_triggers; s ...
- ZOJ-1586 QS Network---最小生成树Prim
题目链接: https://vjudge.net/problem/ZOJ-1586 题目大意: 首先给一个t,代表t个测试样例,再给一个n,表示有n个QS装置,接下来一行是n个QS装置的成本.接下来是 ...
- 南阳OJ-2-括号配对问题---栈的应用
题目链接: http://acm.nyist.edu.cn/JudgeOnline/problem.php?pid=2 题目大意: 有一行括号序列,请你检查这行括号是否配对. 思路: 直接用栈来模拟 ...
- ASP.NET CORE系列【五】webapi整理以及RESTful风格化
介绍 什么是RESTful? 这里不多做赘述,详情请百度! 哈哈,本来还想巴拉巴拉介绍一些webapi, RESTful的, 还是算了,咱们直接上干货!(原因是懒!哈哈) 使用 以前使用过mvc的人 ...
- springcloud分布式事务终极探讨
2018阿里云全部产品优惠券(好东东,强烈推荐)领取地址:https://promotion.aliyun.com/ntms/act/ambassador/sharetouser.html?userC ...
- Java:扩展后的赋值运算符(带强转功能)
扩展后的赋值运算符,即 +=,-=,*=,/=,%=,&=,|=,^=,<<=,>>=,>>>=. 代码实例一: byte a=5; a=a+5; 此 ...