论Java中的抽象类与接口
抽象类和抽象方法
定义
- 抽象方法和抽象类都必须被abstract关键字修饰。
 - 抽象——abstract,抽象类的方法不一定是抽象的,但抽象方法出现的类一定是抽象类。
 
//抽象方法,没有方法体(即没有{}),只有声明
abstract void f();
- 最重要的是:抽象类,抽象既不是真的,所以,抽象类不可以实例化。但可以通过子类的实例化来使用
 
/**
 * @author yhy
 * 用来完成9.2的练习
 * 验证抽象类与抽象方法的使用
 */
public class YanZheng {
    public static void main(String[] args) {
//      不能被实例化,抽象类,会报错
//        ChouXiang chouxi = new ChouXiang() ;
//        可以实例child类
//        即通过继承其子类来实现不能继承抽象类
        Child test = new Child();
    }
}
abstract class AbstractChouXiang{
    /**
     * 构造函数
     */
    AbstractChouXiang() {
    }
    /**
     * 定义一个抽象类的抽象方法
     */
    abstract void chouxiang();
}
class Child extends AbstractChouXiang{
   Child(){
       System.out.println("实例时候就打印出来");
   }
    /**
     * 注意这里不是abstract就不要讲方法定义为abstract
     */
    @Override
    void  chouxiang(){
         System.out.println("继承抽象类");
    }
}
- 子类可以不是抽象类,但要实现抽象父类中的所有抽象方法,否则必须定义为abstract类型。(下面的代码中,我将其子类的重写方法注释掉之后,就会报错must be declared abstract or implentment abstract method)
 

与普通类的区别以及注意点:
抽象类也是可以与普通类那样,可以直接extends,区别在于抽象类不能直接实例化,可以通过实例化其子类,通过子类重写方法实现等——设置抽象方法就是让子类来实现的,否则毫无意义。
与普通方法的区别
抽象方法和空方法体的方法不是同一个概念。例如,public abstract void test();是一个抽象方法,它根本没方法体,即方法定义后面没有一对花括号;但public void test(){}方法是一个普通方法,它已经定义了方法体,只是方法体为空,即它的方法体什么也不做,因此这个方法不可使用abstract来修饰。——疯狂的Java讲义
abstract不能用于修饰Field,不能用于修饰局部变量,即没有抽象变量、没有抽象Field等说法;abstract也不能用于修饰构造器,没有抽象构造器,抽象类里定义的构造器只能是普通构造器。
抽象类的作用
《thinking in java》
抽象类是普通的类与接口之间的一种中庸之道。
抽象方法、抽象类可以使类的抽象性明确起来,告诉用户和编译器怎么使用它们;
同时,抽象类是很好的重构工具,在后期的工作中,可以实现重用性。
体现一种模板的效果,从一群相似的子类提炼出一个抽象类的感觉一样,提供了一种规范,子类可以在其原来的基础上进行扩展。
抽象父类可以只定义需要使用的某些方法,把不能实现的部分抽象成抽象方法,就是一中留给下一代去实现,一开始没有能力去实现,那可就给厉害的人去做,留给其子类去实现。
接口
定义
- 特殊的“抽象类”——接口(interface):比抽象类更加抽象的是接口,在接口中所有的方法都是抽象的。就不能像上面的抽象类一样还可以有普通方法。
 
//省略public就变为默认级别,只能在当前包所访问
public interface Figure {
//接口中静态成员变量
String name = "几何图形";//省略public static final
// 绘制几何图形方法
void onDraw(); //省略public 这里是抽象方法
}
- Java中可以implements多个接口,多继承的含义便是接入多个接口(继承只能单继承)
 - 一个类可以实现一个或多个接口,继承使用extends关键字(但接口只能继承接口),实现则使用implements关键字。
 
示例
- JieKou.java
 
import java.text.SimpleDateFormat;
/**
 * @author yhy
 * 这个是实现接口定义的代码,在其它地方去调用
 * 这里的接口不用public的话,其它的包就访问不了
 */
public interface JieKou {
//   定义了两个常量
    /**
     * 这里定义一个df变量来获取当前时间
     */
    SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");//设置日期格式
    String AUTHOR = "yhycoder";
    /**
     * 定义一个接口的方法
     * 这里的public是多余的,在接口里面是自动为public
     */
    /*public*/ void print();
}
- 使用接口的Java代码
 
//访问了其它包的接口,就是下面这个地址
import music.daima.ebook.JieKou;
import java.util.Date;
public class UseInterfaces {
    public static void main(String[] args) {
//实例化using类,实现查看代码的运行情况
        Using Shuchu = new Using();
        Shuchu.print();
    }
}
/**
 * 这里是接口继承接口
 */
interface Jiekou2 extends JieKou{
    String num = "接口2";
}
/**
 * 这里是Using类实现了JieKou和Jiekou2接口,逗号隔开
 */
class Using implements JieKou,Jiekou2 {
    /**
     * 重写了方法,调用接口定义的常量
     */
    @Override
    public void print() {
        System.out.println(AUTHOR+"在新的包里面使用接口的时间:"+df.format(new Date())+" 同时还有"+num);
    }
}
注意
- 接口与抽象类一样都不能被实例化
 - 实现接口时接口中原有的抽象方法在实现类中必须实现。默认方法可以根据需要有选择实现(覆盖)。静态方法不需要实现,实现类中不能拥有接口中的静态方法。(Java 8之后)
 
//InterfaceA.java文件,定义一个接口
public interface InterfaceA {
//抽象方法
    void methodA();
	String methodB();
// 默认方法
	default int methodC() {
	return "6666";
	}
// 默认方法
	default String methodD() {
	return "这是默认方法";
	}
// 静态方法
	static double methodE() {
	return 0.0;
	}
}
实现接口代码
import xxxx.InterfaceA;
public class ABC implements InterfaceA {
	//重写
    @Override
	public void methodA() {
	}
	@Override
	public String methodB() {
	return "实现methodB方法...";
	}
    //重写覆盖,根据自己的需要来。
	@Override
	public int methodC() {
	return 500;
	}
}
//实现类中不能有接口中的静态方法,最后一行
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());
    }
}
作用
- 规范,在分配不同人的任务时,接口就像是总纲一样,告诉大家去实现哪些功能模块等。(命名规范都有限制到)
 
最后:接口与抽象类的异同
不同
接口interface,实现接口则使用implements;抽象类abstract
抽象类可以有普通方法。Java 8 之前接口中只有抽象方法,而 Java 8 之后接口中也可以声明具体方法,具体方法通过声明默认方法实现。
接口可以继承多个,而抽象类不可以。
和类继承相似,子接口扩展某个父接口,将会获得父接口里定义的所有抽象方法、常量Field、内部类和枚举类定义。
实现父接口的所有:一个类实现了一个或多个接口之后,这个类必须完全实现这些接口里所定义的全部抽象方法(也就是重写这些抽象方法);否则,该类将保留从父接口那里继承到的抽象方法,该类也必须定义成抽象类。
接口定义的是一种规范,因此接口里不能包含构造器和初始化块定义。接口里可以包含Field(只能是常量)、方法(只能是抽象实例方法)、内部类(包括内部接口、枚举)定义。但抽象类与普通类一样,可以有构造器,初始化模块等。
接口只有常量——接口中不能有实例成员变量,接口所声明的成员变量全部是静态常量,即便是变量不加 public static final 修饰符也是静态常量。抽象类与普通类一样各种形式的成员变量都可以声明。

相同
- 都不能直接实例化来使用。
 - 接口和抽象类都可以包含抽象方法,实现接口或继承抽象类的普通子类都必须实现这些抽象方法。
 
使用场景
想要多重继承的时候——接口(功能性强,规范性)
想要底层基础功能模块不断改变——抽象类(模板设计)
感谢
才疏学浅,不对的地方多多指教!
借鉴
论Java中的抽象类与接口的更多相关文章
- Java 中,抽象类与接口之间有什么不同?
		
Java 中,抽象类和接口有很多不同之处,但是最重要的一个是 Java 中限制一个 类只能继承一个类,但是可以实现多个接口.抽象类可以很好的定义一个家族类 的默认行为,而接口能更好的定义类型,有助于后 ...
 - java中的抽象类和接口
		
抽象类和接口本身让面向对象真正实现,一个好的系统可以让抽象类或者接口实现多次复用,如果出现了集成具体类那么肯定是有问题的. 抽象类和接口很相似,很多时候好像功能可以混用,java设计者赋予了很多不一样 ...
 - Java 中的抽象类及接口
		
抽象类使用 abstract 关键字修饰,该类即为抽象类. 抽象类的作用: 1.某些情况下,父类约束子类必须包含哪些方法,但不知道子类如何去实现这些方法. 2.可以从多个具有相同特征的类中抽象出一个抽 ...
 - JAVA中的抽象类与接口
		
抽象类 abstract class 包含抽象方法的类,叫抽象类.而抽象的概念就是抽象出共同属性:成员变量和方法.所以抽象类可以有private等多种权限的成员变量和非abstract的成员方法.当然 ...
 - 辨异 —— Java 中的抽象类和接口
		
接口优于抽象类.-- <Effective Java>(Item 18) 0. 语法区别 抽象类允许给出某些方法的实现,接口不允许: 为了实现由抽象类定义的类型(type),类必须成为抽象 ...
 - java基础(四)-----抽象类与接口
		
抽象类与接口是java语言中对抽象概念进行定义的两种机制,正是由于他们的存在才赋予java强大的面向对象的能力.他们两者之间对抽象概念的支持有很大的相似,甚至可以互换,但是也有区别. 一.抽象类 我们 ...
 - Java学习笔记二十七:Java中的抽象类
		
Java中的抽象类 一:Java抽象类: 在面向对象的概念中,所有的对象都是通过类来描绘的,但是反过来,并不是所有的类都是用来描绘对象的,如果一个类中没有包含足够的信息来描绘一个具体的对象,这样的类就 ...
 - Java基础之抽象类与接口
		
Java基础之抽象类与接口 对于面向对象编程来说,抽象是它的一大特征之一.在Java中,可以通过两种形式来体现OOP的抽象:接口和抽象类.这两者有太多相似的地方,又有太多不同的地方.很多人在初学的时候 ...
 - C++中的抽象类和接口
		
1,在 C++ 语言中,并不直接支持面向对象中的抽象类和接口概念,但是 C++ 语言 却可以间接实现这些概念: 2,什么是抽象类: 1,面向对象中的抽象(其实就是分类)概念: 1,在进行面向对象分析时 ...
 
随机推荐
- ACM感想、
			
从15年10月开始搞ACM,到如今的16年4月底,已经接近半年了. 大一下学期开始ACM组的人员就不断减少,有的因为坚持不下,有的因为不喜欢,有的没留下理由就走了, 瞬间感觉实验室空了很多很多.现在常 ...
 - Python __call__详解
			
可以调用的对象 关于 __call__ 方法,不得不先提到一个概念,就是可调用对象(callable),我们平时自定义的函数.内置函数和类都属于可调用对象,但凡是可以把一对括号()应用到某个对象身上都 ...
 - 2018-6-11-WPF-Frame-的-DataContext-不能被-Page-继承
			
title author date CreateTime categories WPF Frame 的 DataContext 不能被 Page 继承 lindexi 2018-06-11 10:48 ...
 - Python--day64--找到作者关联的所有书籍对象、ORM多对多关联查询的原理
			
找到当前作者关联的所有书籍对象: ORM多对多关联查询的原理: 编辑作者:
 - 洛谷P1809 过河问题 经典贪心问题
			
作者:zifeiy 标签:贪心 题目链接:https://www.luogu.org/problem/P1809 我们假设第 \(i\) 个人过河的耗时是 \(t[i]\) ,并且 \(t[i]\) ...
 - 【codeforces 761D】Dasha and Very Difficult Problem
			
time limit per test2 seconds memory limit per test256 megabytes inputstandard input outputstandard o ...
 - 原生Js 实现等比缩放页面
			
针对1920*1080 分配率缩放 window.addEventListener('load', adaptation); window.addEventListener('resize', ada ...
 - 【js】Vue 2.5.1 源码学习 (八)响应式入口observe
			
大体思路(七) 本节内容: deps 依赖收集的数组对象 => Dep 构造函数 /** ==> observe() * var ob * ==> if --isObject * = ...
 - H3C CIDR
 - JS 手札记
			
addEventListener中的事件如果移除(removeEventListener)的话不能在事件中执行bind(this)否则会移除无效! // selectCurrent() // copy ...