java中抽象类和接口的相同和不同点介绍
作者:白了又了白1992 发布时间:2021-05-30 04:53:31
前言
本文简单介绍抽象类,接口以及它们的异同点,另附简单的代码举例。
一、抽象类是什么?
在 Java 语言中使用 abstract class 来定义抽象类。抽象类是不能被实例化的,即不能用new关键字来实例化对象。包含抽象方法的一定是抽象类,但抽象类不一定包含抽象方法。如果一个子类实现了父类(抽象类)的所有抽象方法,那么该子类可以不必是抽象类,否则就是抽象类。抽象类中的抽象方法的修饰符只能为public或者protected。
为什么要在抽象类中定义抽象方法呢?下面我们来看一个代码实例:
package example;
import java.util.Date;
public abstract class GeometricObject {
private String color="white";
private boolean filled;
private java.util.Date dateCreated;
protected GeometricObject() {
dateCreated=new java.util.Date();
}//构建默认的几何对象
protected GeometricObject(String color,boolean filled) {
dateCreated=new java.util.Date();
this.color=color;
this.filled=filled;
}
public String getColor() {
return color;
}//返回颜色
public void setColor(String color) {
this.color=color;
}//设置新的颜色
public boolean getFilled() {
return filled;
}
public void setFilled(boolean filled) {
this.filled=filled;
}
public java.util.Date getDateCreated(){
return dateCreated;
}
public abstract double getArea();//抽象方法定义得到面积
public abstract double getPerimeter();//抽象方法得到周长
}
我们假设自定义的Circle类和Rectangle类均为GeometricObject的子类,且均有求各自面积的getArea()方法和各自周长的getPerimeter()方法,但这些方法并不能在父类中定义,因为具体的计算方法要取决于几何对象的具体类型。所以采取抽象方法进行定义,这样可以保证父类中的抽象方法可以在子类中被重写。
抽象类只有定义没有实现。
下面是抽象类值得注意的几点:
1.抽象方法不能包含在非抽象类中。如果抽象父类的子类不能实现所有的抽象方法,那么子类也必须定义为抽象的。也就是说,在继承自抽象类的非抽象子类中,必须实现所有的抽象方法。还要注意到,抽象方法是静态的。
2.抽象类不能使用new操作符来初始化。但仍然可以定义它的构造方法,这个构造方法在它的子类的构造方法中进行调用。
3.包含抽象方法的类必须是抽象的。但是是可以定义一个不包含抽象方法的抽象类,这个抽象类用于作为定义新子类的基类。
4.即使子类的父类是具体的,这个子类也可以是抽象的。
5.不能使用new操作符从一个抽象类创建一个实例,但是抽象类可以用做一种数据类型。
例如:GeometricObject[] objects=new GeometricObject[10];是正确的,。
6.子类可以重写父类的方法并将它定义为抽象的,这虽然不常见,但很适用于当父类的方法实现在子类中变得无效时的情况。
二、接口是什么?
接口是一种与类很相似的结构,用于为对象定义共同操作,但它的目的是指明相关或者不相关类的对象的共同行为。
Java中接口使用interface关键字修饰。接口是抽象方法的集合。如果一个类实现了某个接口,那么它就继承了这个接口的抽象方法。
类和接口之间的关系称为接口继承。
下面是简单的代码举例:
package example;
public class TestEdible {
public static void main(String[] args) {
// TODO Auto-generated method stub
Object[] objects= {new Tiger(),new Chicken(),new Apple()};
//创建由Tiger,Chicken和Apple类型的三个对象构成的数组
for(int i=0;i<objects.length;i++) {
if(objects[i] instanceof Edible)
System.out.println(((Edible)objects[i]).howToEat());
if(objects[i] instanceof Animal) {
System.out.println(((Animal)objects[i]).sound());
}
}//如果可食用,则调用howToEat方法,如果是动物,则调用sound方法
}
}
abstract class Animal{
private double weight;
public double getWeight() {
return weight;
}
public void setWeight(double weight) {
this.weight=weight;
}
public abstract String sound();//返回动物的声音,是抽象方法,被具体的animal类实现
}
class Chicken extends Animal implements Edible{
@Override
public String howToEat() {
return "Chicken: Fry it";
}//实现了Edible接口。当一个类实现接口时,该类实现定义在接口中的所有方法。
@Override
public String sound() {
return "Chicken: cock-a-doodle-doo";
}
}
class Tiger extends Animal{
@Override
public String sound() {
return "Tiger:RROOAARR";
}
}
abstract class Fruit implements Edible{
}//Fruit类实现了Edible,但没有实现howToEat方法,所以它必须定义为abstract。
class Apple extends Fruit{
@Override
public String howToEat() {
return "Apple:Make apple cider";
}
}//Fruit的具体子类必须实现howToEat方法,所以Apple类实现了howToEat方法。
需要注意的是,接口中所有的数据域都是public static final,而且所有方法都是public abstract,但Java允许忽略这些修饰符。
即下面的两个代码是等价的:
public interface T {
public static final int K=1;
public abstract void p();
}
public interface T {
int K=1;
void p();
}
尽管public修饰符对于定义在接口中的方法可以省略,但是在子类实现中方法必须定义为public。
三、抽象类和接口的异同点
一个类可以实现多个接口,但是只能继承一个父类。
首先我们先来列举个表格简单讲述一下抽象类和接口之间的异同点:
比较点 | 抽象类 | 接口 |
关键字 | abstract class | interface |
字段 | 无限制 | 变量必须是public static final |
方法 | 既可以含普通方法,又可以含抽象方法,无限制 | 只能含抽象方法,且必须是public的 |
继承/实现 | 只能被类或抽象类继承 | 既可以被接口继承,也能被类或抽象类实现 |
多重继承 | 不支持 | 可以继承多个父接口 |
java只允许为类的继承做单一继承,但是允许使用接口做多重继承。例如:
public class NewClass extends BaseClass implements Interface1,...,InterfaceN{
}
利用extends,接口可以继承其他接口,这样的接口被称为子接口。例如:下面的代码中,NewInterface是Interface1,...,InterfaceN的子接口。
public interface NewInterface extends Interface1,...,InterfaceN{
}
一个实现NewInterface的类必须实现在NewInterface,Interface1,...,InterfaceN中定义的抽象方法。接口可以继承其他接口但不能继承类。一个类可以继承它的父类同时实现多个接口。
所有的类都有一个共同的根类Object,但是接口并没有共同的根。与类相似,接口同样也可以定义一种类型。一个接口类型的变量可以引用任何实现该接口的类的实例。如果一个类实现了一个接口,那么这个接口就类似于该类的一个父类。可以将接口当作一种数据类型使用,将接口类型的变量转换为它的子类,反过来同样可以。
通常情况下,使用接口的频率更高,因为接口比类更加灵活,不用使所有东西都属于同一个类型的类。
下面进行一个简单的代码举例:
abstract class Animal{
public abstract String howToEat();
}//在Animal中定义howToEat方法
//Animal的两个子类定义如下
class Chicken extends Animal{
@Override
public String howToEat() {
return "Chicken: Fry it";
}
}
class Duck extends Animal{
@Override
public String howToEat() {
return "Roast it";
}
}
public static void main(String[] args) {
Animal animal = new Chicken();
eat(animal);
Animal animal = new Duck();
eat(animal);
}
public static void eat(Animal animal){
System.out.println(animal.howToEat());
}
假设给定这个继承结构,java在调用方法时可以根据对象动态地决定调用具体的howToEat方法。但有一个限制条件,即该子类必须是另一种动物才可以。如果一种动物不可食用,那么此刻再继承Animal类就并不合适了。
限制条件这个问题在接口中可以被解决,例如:
public class Demo{
public static void main(String[] args) {
Edible stuff = new Chicken();
eat(stuff);
Edible stuff = new Broccoli();
eat(stuff);
}
public static void eat(Edible stuff){
System.out.println(stuff.howToEat());
}
}
interface Edible{
public String howToEat();
}
class Chicken implements Edible{
@Override
public String howToEat() {
return "Chicken: Fry it";
}
}
class Broccoli implements Edible{
@Override
public String howToEat() {
return "Broccoli: Stir-fty it";
}
}
定义表示可食用对象的一个类,只须让该类实现Edible接口即可。任何Edible对象都可以被传递以调用HowToEat方法。
来源:https://blog.csdn.net/weixin_51368291/article/details/121875445


猜你喜欢
- 在定位JVM性能问题时可能会遇到内存泄露导致JVM OutOfMemory的情况,在使用Tomcat容器时如果设置了reloadable=”
- 为什么需要互斥量在多任务操作系统中,同时运行的多个任务可能都需要使用同一种资源。这个过程有点类似于,公司部门里,我在使用着打印机打印东西的同
- /// <summary> /// 计算本周起始日期(礼拜一的日期) /// </summary&
- 前言前面几篇我们简单的复习了一下自定义 View 的测量与绘制,并且回顾了常见的一些事件的处理方式。那么如果我们想自定义 ViewGroup
- 在Android Studio 3.0中一旦我们创建了一个项目,一个名为mipmap-anydpi-v26自动创建的文件夹在res文件夹下。
- Junit这种老技术,现在又拿出来说,不为别的,某种程度上来说,更是为了要说明它在项目中的重要性。 凭本人的感觉和经验来说,在项目中完全按标
- 案例需求:访问带有验证码的登录页面login.jsp用户输入用户名,密码以及验证码。如果用户名和密码输入有误,跳转登录页面,提示:用户名或密
- 首先我们应该清楚的是JDK1.6和JDK1.7中String类的intern方法还是有差别的: JDK1.6中的int
- 配置Thymeleaf的模板路径众所周知,Thymeleaf的模板文件默认是在项目文件夹的src\main\resources\templa
- 在平时的开发中,我们会经常遇到这样一个需求,要在页面通过一个『导出』按钮把查询出的数据导出到 Excel 表格中。本文即为实现上述需求的一个
- Android WebView 1.首先修改activity.xml中的代码:2.然后MainActivity中的代码:3.最后设置权限:&
- 在观察者模式中有2个要素:一个是被观察对象,另一个是观察者。但被观察对象的状态发生改变会通知观察者。举例:把订阅报纸的人看作是观察者,把报纸
- 1.前言以前没有使用mybatis,可以关闭自动提交,然后做sql操作,对操作进行catch捕获异常,如果没有异常则commit 提交 ,有
- 本文实例讲述了Android开发之获取LayoutInflater对象的方法。分享给大家供大家参考,具体如下:在写Android程序时,有时
- Android 通过Socket 和服务器通讯,是一种比较常用的通讯方式,时间比较紧,说下大致的思路,希望能帮到使用socket 进行通信的
- 最近为公司做的一个Demo里面用到了ScrollView嵌套了GridView和ListView,然而在嵌套的时候我发现GridView和L
- 本文实例讲述了Java数据结构之链表、栈、队列、树的实现方法。分享给大家供大家参考,具体如下:最近无意中翻到一本书,闲来无事写几行代码,实现
- 本文实例讲述了C#实现查杀本地与远程进程的方法。分享给大家供大家参考。具体实现方法如下:using System;using System.
- 前言我们利用printf 函数实现一个在屏幕上弹跳的小球,如图所示。弹跳的小球游戏比较简单、容易入门,也是反弹球消砖块、接金币、台球等很多游
- 最近有个老项目想逐步将新业务的数据放到新的数据库,以前的业务还得连接以前的数据库,于是需要整合多数据源 。多数据源实际上是继承了Abstra