知识点:
抽象类 接口 面向接口开发
一、抽象类的用法
1.1 复习:抽象类构造函数测试
抽象类不可以创建对象,抽象类中的构造函数常在子类对象实例化时使用
package cn.mbh.oop;
/**
*@author MBH:
*@version 创建时间:2022年7月10日 下午6:50:02
*/
/**本类用于测试抽象类的构造函数的用法*/
/**总结:
* 1.抽象类中可以有构造方法
* 2.父类的构造方法要优先于子类的执行
* 2.抽象类中的构造方法存在的目的,不是为了创建抽象类本身的对象,而是为了创建子类对象时使用的
*/
public class Test4_Abstract2 {
public static void main(String[] args) {
//4.创建多态对象进行测试
Animal3 a = new Dog3();
//5.测试抽象类是否可以创建对象?---不能!
//Animal3 a3 = new Animal3();
}
}
//1.创建抽象父类
abstract class Animal3{
//3.创建抽象类的构造函数
public Animal3() {
//如果此处传参,会覆盖无参构造,子类super就报错了
System.out.println("Animal3....构造方法");
}
}
//2.创建子类
class Dog3 extends Animal3{
//5.创建子类无参构造,无参构造默认存在
public Dog3() {
super();//隐藏着super();//先访问父类的构造方法,再执行自己的功能
System.out.println("Dog3...构造方法");
}
}
1.2 练习 1:抽象类成员测试
创建 Java 工程: DAY06
创建包: cn.tedu. oop
创建类: Test1_Abstract3.java
package cn.mbh.oop;
/**
*@author MBH:
*@version 创建时间:2022年7月10日 下午7:19:10
*/
/**本类用于测试抽象类中的成员*/
public class Test1_Abstract {
public static void main(String[] args) {
//4.创建多态对象进行测试
Fruit f = new Apple();
System.out.println(f.sum);//10
//name = "lemon";//常量值补课修改
System.out.println(f.name);//banana
}
}
//1.创建抽象父类--水果类
abstract class Fruit{
/**1.抽象类中可以有成员变量吗?--可以!!!*/
//3.1创建抽象父类成员变量
int sum = 10;
/**2.抽象类中可以有成员常量,但是值无法修改*/
//3.2创建抽象父类成员常量
final String name = "banana";
/**3.抽象类中可以有普通成员方法,也可以都是普通方法*/
/**4.如果一个类中都是普通方法,这个类为什么还要被声明成抽象类呢?
* 原因:抽象类不可以创建对象,如果不想让外界创建本类的对象,可以把普通类声明成一个抽象类
* */
//5.创建普通成员方法*2
public void eat(){
System.out.println("吃啥水果都行");
}
public void eat2(){
System.out.println("吃啥水果都行");
}
/**5.抽象类中可以有抽象方法,一旦类中有抽象方法,这个类必须被声明成抽象类*/
abstract public void clean();
}
/**6.当一个类继承了父类,父类是一个抽象类时
* 子类需要重写抽象父类的所有抽象方法,或者把自己变成一个抽象子类
* */
//解决方案一:抽象子类
//abstract class Apple extends Fruit{ }
//解决方案二:
//2.创建子类 -- 苹果类 :普通子类+重写父类的抽象方法
class Apple extends Fruit{
//子类重写父类的抽象方法clean()
@Override //表示这个方法是一个重写的方法
public void clean() {
System.out.println("苹果还是需要好好洗洗再吃的");
}
}
继承+抽象类 – 接口引入
abstract class Animal{abstract public void eat();abstract public void sleep();}class Dog extends Animal{eat(){......}}class Cat extends Animal{eat(){......}}class Pig extends Animal{eat(){......}}
二、接口
2.1 概念
Java 里面由于不允许多重继承,所以如果要实现多个类的功能,则可以通过实现多个接口来实现。
Java 接口和 Java
抽象类
代表的就是抽象类型,就是我们需要提出的抽象层的具体表现。
OOP
面向对象
的编程,如果要提高程序的复用率,增加程序的可维护性,可扩展性,就必须是面向接口的编程,面向抽象的编程,正确地使用接口、抽象类这些太有用的抽象类型做为 java 结构层次上的顶层。
2.2 接口的格式
interface 接口名 { 代码 … }
2.3 特点
1. 接口中都是抽象方法
2. 通过 interface 关键字来定义接口
3. 通过 implements 让子类来实现接口
4. 可以理解成,接口是一个特殊的抽象类(接口里的方法都是抽象方法)
5. 接口突破了 java 的单继承的局限性
6. 接口和类之间可以多实现,接口和接口之间可以多继承
7. 接口是对外暴露的规则,是一套开发规范
8. 接口提高了程序的功能扩展,降低了耦合性
2.4 练习 2-1:创建接口
创建 Java 工程: DAY06
创建包: cn.tedu. inter
创建接口: Inter.java
package cn.mbh.inter;
/**
*@author MBH:
*@version 创建时间:2022年7月10日 下午7:32:42
*/
/**本接口是创建接口测试*/
/**1.通过interface关键字来定义接口*/
public interface Inter {
/**2.接口中可以有普通方法吗?--不可以!!!*/
//public void eat() {}
/**3.接口中可以有抽象方法吗?--可以,接口中的方法都是抽象方法!!!*/
public void eat();
public void play();
}
2.5 练习 2-2:创建接口实现类
创建 Java 工程: DAY06
创建包: cn.tedu. inter
创建接口实现类: InterImpl.java
package cn.mbh.inter;
/**
*@author MBH:
*@version 创建时间:2022年7月10日 下午7:35:42
*/
/**本类作为Inter接口的实现类*/
/**1.实现类如果想用接口的功能,要和接口发生实现关系,通过关键字implements*/
/**2.1方案一:实现类实现了接口以后,可以把自己变成一个抽象子类*/
//abstract public class InterImpl implements Inter{ }
/**2.2方案二:实现类实现了接口后,还可以重写接口中所有抽象方法*/
public class InterImpl implements Inter{
@Override//标记重写了父接口的方法
public void eat() {
System.out.println("吃火锅");
}
@Override//标记重写了父接口的方法
public void play() {
System.out.println("玩代码");
}
}
2.6 练习 2-3:创建接口测试类
创建 Java 工程: DAY06
创建包: cn.tedu. inter
创建实现类测试类: InterTests.java
package cn.mbh.inter;
/**
*@author MBH:
*@version 创建时间:2022年7月10日 下午7:39:34
*/
/**本类用于测试接口实现类*/
public class InterTests {
//1.创建入口main()
public static void main(String[] args) {
//2.创建多态对象进行测试
Inter inter = new InterImpl();
inter.eat();
inter.play();
//3.接口可以创建对象吗?--不行!!!
//Inter inter = new Inter();
}
}
三、接口的用法
3.1 练习 3-1:接口之构造方法
创建 Java 工程: DAY06
创建包: cn.tedu. inter
创建类: Test5_UseInter.java
package cn.mbh.inter;
/**
*@author MBH:
*@version 创建时间:2022年7月10日 下午7:42:57
*/
/**本类用于测试接口的使用*/
public class Test5_UseInter {
public static void main(String[] args) {
//3.在main()创建多态对象进行测试
/**问题:子类创建对象时,会自动调用父类的构造方法,但现在的父级是个接口
* 接口没有构造方法,那子类中super()调用的是什么呢?
* 结论1:子类默认继承了顶级父类Object,super()会自动调用Object的无参构造
* */
Inter2 in = new Inter2Impl();
}
}
//1.创建接口
interface Inter2{
/**1.接口里有构造方法吗?--没有!!!*/
//public Inter2() {}
}
//2.创建接口的实现类
class Inter2Impl implements Inter2{
public Inter2Impl() {
super();//默认先调用顶级父类Object的无参构造方法
System.out.println("Inter2Impl的无参构造");
}
}
3.2 练习 3-2:接口之成员变量
package cn.mbh.inter;
/**
*@author MBH:
*@version 创建时间:2022年7月10日 下午7:42:57
*/
/**本类用于测试接口的使用*/
public class Test5_UseInter {
public static void main(String[] args) {
//3.在main()创建多态对象进行测试
/**问题:子类创建对象时,会自动调用父类的构造方法,但现在的父级是个接口
* 接口没有构造方法,那子类中super()调用的是什么呢?
* 结论1:子类默认继承了顶级父类Object,super()会自动调用Object的无参构造
* */
Inter2 in = new Inter2Impl();
System.out.println(in.age);
/**结论2.1接口中的变量都是静态变量,可以通过接口名直接调用*/
System.out.println(Inter2.age);
/**结论2.2接口中的变量都是常量,不能被重新赋值*/
//Inter2.age = 20;
}
}
//1.创建接口
interface Inter2{
/**1.接口里有构造方法吗?--没有!!!*/
//public Inter2() {}
/**2.接口里可以有成员变量吗?--没有!!!*/
int age = 10;//静态常量,实际上:final static int age =10;
}
//2.创建接口的实现类
class Inter2Impl implements Inter2{
public Inter2Impl() {
super();//默认先调用顶级父类Object的无参构造方法
System.out.println("Inter2Impl的无参构造");
}
}
3.3 练习 3-3:接口之成员方法
package cn.mbh.inter;
/**
*@author MBH:
*@version 创建时间:2022年7月10日 下午7:42:57
*/
/**本类用于测试接口的使用*/
public class Test5_UseInter {
public static void main(String[] args) {
//3.在main()创建多态对象进行测试
/**问题:子类创建对象时,会自动调用父类的构造方法,但现在的父级是个接口
* 接口没有构造方法,那子类中super()调用的是什么呢?
* 结论1:子类默认继承了顶级父类Object,super()会自动调用Object的无参构造
* */
Inter2 in = new Inter2Impl();
System.out.println(in.age);
/**结论2.1接口中的变量都是静态变量,可以通过接口名直接调用*/
System.out.println(Inter2.age);
/**结论2.2接口中的变量都是常量,不能被重新赋值*/
//Inter2.age = 20;
}
}
//1.创建接口
interface Inter2{
/**1.接口里有构造方法吗?--没有!!!*/
//public Inter2() {}
/**2.接口里可以有成员变量吗?--没有!!!*/
int age = 10;//静态常量,实际上:final static int age =10;
/**3.接口里可以有抽象方法*/
abstract public void eat2();
void eat();//可以简写--会自动拼接public abstract
}
//2.创建接口的实现类
class Inter2Impl implements Inter2{
public Inter2Impl() {
super();//默认先调用顶级父类Object的无参构造方法
System.out.println("Inter2Impl的无参构造");
}
/**4.如果接口中添加了抽象方法,实现类中需要添加所有未实现的抽象方法*/
@Override
public void eat2() {
}
@Override
public void eat() {
}
}
四、程序设计
4.1 分析老师示例-面向抽象编程
具体事物:实习老师 正式老师共性:讲课,备课
创建 Java 工程: DAY06
创建包: cn.tedu. design
创建类: Test6_Design.java
/**
*@author MBH:
*@version 创建时间:2022年7月10日 下午7:51:40
*/
public class Test6_design {
}
//生活中的事物--java类
//特征--属性 行为--方法
//把所有的共性内容向上抽取形成父类
/**2.类中含有抽象方法,这个类必须变成抽象类*/
abstract class Teacher{
//讲课
/**1.没有方法体的方法叫做抽象方法*/
abstract public void teach();
//备课
public void ready() {
System.out.println("正在备课");
}
}
//实习老师
class SXTeacher extends Teacher{
//讲课
@Override
public void teach() {
System.out.println("讲课");
}
//备课
@Override
public void ready() {
System.out.println("正在备课");
}
}
//正式老师--基础加强+框架加强+高新技术
class ZSTeacher extends Teacher{
//讲课
@Override
public void teach() {
System.out.println("正在讲课...基础加强+框架加强+高新技术");
}
}
//Scala大数据老师--Scala+Spark
abstract class ScalaTeacher extends Teacher{
}
4.2 分析老师示例-面向接口编程
具体事物:培优班老师, 高手班老师共性:讲课,备课
创建 Java 工程: DAY06
创建包: cn.tedu. design
创建类: Test7_DesignInter.java
package cn.mbh.design;
/**
*@author MBH:
*@version 创建时间:2022年7月10日 下午8:04:20
*/
/**本类用于设计老师类,面向接口编程*/
public class Test7_DesignInter {
}
//抽取共性,形成抽象层--体现接口--定义规则
/**1.通过interface关键字定义接口*/
interface Teacher2{
/**2.接口中的方法都是抽象方法*/
/**3.接口中的方法可以简写--自动拼接public abstract*/
//讲课方法
void teach();
//备课方法
void ready();
}
/**4.实现类要使用接口的功能,需要发生实现关系*/
//实习老师
class SXTeacher2 implements Teacher2{
@Override
public void teach() {
}
@Override
public void ready() {
System.out.println("备课电商项目");
}
}
//正式老师
abstract class ZSTeacher2 implements Teacher2{ }
六、接口的多继承多实现
6.1 代码示例
创建 Java 工程: DAY06
创建包: cn.tedu. design
创建类: Test8.java
package cn.mbh.design;
/**
*@author MBH:
*@version 创建时间:2022年7月10日 下午8:07:43
*/
public class Test8 {
public static void main(String[] args) {
//创建多态对象进行测试
Inter1 in = new InterImpl();
in.delete();
in.save();
//in.update();//运行看右边(子类),不能调用子类特有的方法
}
}
//1.创建接口1
interface Inter1{
void save();
void delete();
}
//2.创建接口2
interface Inter2{
void update();
void find();
}
//4.创建接口3测试接口间的继承关系
/**2.接口之间可以发生继承关系,而且可以多继承,用逗号隔开*/
interface Inter3 extends Inter1,Inter2{}
/**3.实现类,可以继承的时候同时多实现
* 注意:由于java是单继承 多实现,所以先写继承,再写实现
* * */
//class InterImpl extends Object implements inter1,inter2{}
/**1.接口和实现类之间可以发生实现关系,通过implements关键字来完成
* 而且可以多实现(同时实现多个接口),多个接口之间用逗号隔开
* */
//3.创建接口的实现类
class InterImpl implements Inter1,Inter2{
@Override
public void update() {
System.out.println("更新中...");
}
@Override
public void find() {
System.out.println("查找中...");
}
@Override
public void save() {
System.out.println("保存中...");
}
@Override
public void delete() {
System.out.println("删除中....");
}
}
七、总结
1. 类与类的关系--继承关系 , 只支持单继承--比如 ,A 是子类 B 是父类 ,A 具备 B 所有的功能 ( 除了父类的私有资源 )--子类如果要修改原有功能 , 需要重写 ( 方法声明与父类一致 + 权限修饰符 >= 父类修饰符 )2. 类和接口的关系--实现关系 . 可以单实现 , 也可以多实现--class A implements B,C{}--其中 A 是实现类 ,B 和 C 是接口 ,A 拥有 BC 接口的所有功能 , 只是需要进行方法的重写 , 否则 A 就是抽象类3. 接口与接口的关系--是继承关系 , 可以单继承 , 也可以多继承--interface A extends B,C{}--其中 ABC 都是接口 ,A 是子接口 , 具有 BC 接口的所有功能 ( 抽象方法 )--class X implements A{}--X实现类需要重写 ABC 接口的所有方法 , 否则就是抽象类--class A extends B implements C,D{}--其中 A 是实现类 , 也是 B 的子类 , 同时拥有 CD 接口的所有功能--这时 A 需要重写 CD 接口里的所有抽象方法4. 抽象类与接口的区别--抽象类是一个特殊的类 , 特殊在 , 抽象类中可以包含没有方法体的方法 ( 抽象方法 )--接口是一个特殊的抽象类 , 特殊在 , 接口里的都是抽象方法 , 没有普通方法--接口会为方法自动拼接 public abstract, 还会为变量自动拼接 public final static--抽象类可以有构造方法 -- 用来给子类创建对象 , 接口没有抽象方法--抽象类和接口都不能实例化 ( 创建对象 )--抽象类可以有普通方法 , 接口都是抽象方法
八、访问控制符
用来控制一个类,或者类中的成员的访问范围。
类 | 包 | 子类 | 任意 | |
public | √ | √ | √ | √ |
protected | √ | √ | √ | |
default
| √ | √ | ||
private | √ |
TIPS:default 是表示不写修饰符,默认,如果写 default 单词来修饰会报错
九、拓展
9.1 abstract 注意事项
抽象方法要求子类继承后必须重写。那么,
abstract
关键字不可以和哪些关键字一起使用呢?以下关键字, 在抽象类中。用是可以用的,只是没有意义了。
1) private:被私有化后,子类无法重写,与
abstract
相违背。
2) static:静态的,优先于对象存在。而
abstract
是对象间的关系,存在加载顺序问题。
3)final:被
final
修饰后,无法重写,与
abstract
相违背。
9.2 接口和抽象类的区别
1、抽象类和接口都不能直接实例化,如果要实例化,抽象类变量必须指向实现所有抽象方法的子类对象,接口变量必须指向实现所有接口方法的类对象。
2、抽象类要被子类继承,接口要被类实现。
3、接口只能做方法申明,抽象类中可以做方法申明,也可以做方法实现
4、接口里定义的变量只能是公共的静态的常量,抽象类中的变量是普通变量。
5、抽象类里的抽象方法必须全部被子类所实现,如果子类不能全部实现父类抽象方法,那么该子类只能是抽 象类。同样,一个实现接口的时候,如不能全部实现接口方法,那么该类也只能为抽象类。
6、抽象方法只能申明,不能实现,接口是设计的结果 ,抽象类是重构的结果
7、抽象类里可以没有抽象方法
8、如果一个类里有抽象方法,那么这个类只能是抽象类
9、抽象方法要被实现,所以不能是静态的,也不能是私有的。
10、接口可继承接口,并可多继承接口,但类只能单根继承。