首页 > 技术文章 > Educoder-Java程序开发基础-封装、继承和多态的综合练习

junun 2021-07-22 17:11 原文

第1关:封装、继承和多态进阶(一)

package case1;

import java.util.Scanner;

public class Task1 {
	public static void main(String[] args) {
		Scanner sc = new Scanner(System.in);
		String dogName = sc.next();		//输入各属性
		String dogSex = sc.next();
		String dogColor = sc.next();
		String catName = sc.next();
		String catSex = sc.next();
		double catWeight = sc.nextDouble();
		// 通过有参构造函数实例化Dog类对象dog
		// dog调用talk()方法
		// dog调用eat()方法
		/********* begin *********/
		Dog dog = new Dog(dogName,dogSex,dogColor);
		dog.talk();
		dog.eat();
		/********* end *********/
		// 通过有参构造函数实例化Cat类对象cat
		// cat调用talk()方法
		// cat调用eat()方法
		/********* begin *********/
		Cat cat=new Cat(catName,catSex,catWeight);
		cat.talk();
		cat.eat();
		/********* end *********/
	}
}

// 抽象类Pet 封装属性name和sex
// 构造函数初始化name和sex
// 声明抽象方法talk()
// 声明抽象方法eat()
abstract class Pet {
	/********* begin *********/
	private String name; 	//封装属性用private
	private String sex;
	public Pet(String name,String sex){
		this.name=name;
		this.sex=sex;
	}
	abstract void talk();
	abstract void eat();
	    public String getName(){ 	//用get方法获取属性
        return name;
    }
    public String getSex(){
        return sex;
    }
	/********* end *********/
}

// Dog类继承自Pet类 封装属性color
// 构造函数初始化name、sex和color
// 实现自己的talk()方法和eat()方法
// talk()输出'名称:name,性别:sex,颜色:color,汪汪叫'
// eat()输出'name吃骨头'
class Dog extends Pet {
	/********* begin *********/
	private String color;
	public Dog(String name,String sex,String color){
		super(name,sex); 	//用super初始化属性
		this.color=color;
	}
	public void talk(){
		System.out.println("名称:"+super.getName()+",性别:"+super.getSex()+",颜色:"+color+",汪汪叫");
	} 	//super.方法名获取属性
	public void eat(){
		System.out.println(super.getName()+"吃骨头!");
	}
	/********* end *********/
}

// Cat类继承自Pet类 封装属性weight
// 构造函数初始化name、sex和weight
// 实现自己的talk()方法和eat()方法
// talk()输出'名称:name,性别:sex,体重:weight kg,喵喵叫'
// eat()输出'name吃鱼'
class Cat extends Pet {
	/********* begin *********/
	private double weight;
	public Cat(String name,String sex,double weight){
		super(name,sex);
		this.weight=weight;
	}
	public void talk(){
		System.out.println("名称:"+super.getName()+",性别:"+super.getSex()+",体重:"+weight+"kg,喵喵叫");
	}
	public void eat(){
		System.out.println(super.getName()+"吃鱼!");
	}
	/********* end *********/
}

第2关:封装、继承和多态进阶(二)

相关知识

为了完成本关任务,你需要掌握:1.重写和重载;2.abstract(抽象类)和interface(接口);3.final关键字;4.static关键字;5.多态。

重写和重载

方法重载(overload):

  • 必须是同一个类;
  • 方法名(也可以叫函数)一样;
  • 参数类型不一样或参数数量或顺序不一样;
  • 不能通过返回值来判断重载。

方法的重写(override)子类重写了父类的同名方法,两同两小一大原则:

  • 方法名相同,参数类型相同;
  • 子类返回类型是父类返回类型的子类;
  • 子类抛出异常小于等于父类方法抛出异常;
  • 子类访问权限大于等于父类方法访问权限。

img

  • 在重写中,运用的是动态单分配,根据new的类型确定对象,从而确定调用的方法;

  • 在重载中,运用的是静态多分配,根据静态类型确定对象,不能根据new的类型确定调用方法;

  • 多态中,Father f = new Son()

    成员变量:编译运行参考左边; 成员函数:编译看左边,运行看右边; 静态函数:编译运行看左边。

abstract(抽象类)和interface(接口)

抽象类

  • abstract修饰的类表示抽象类,抽象类位于继承树的抽象层,抽象类不能被实例化。
  • abstract修饰的方法表示抽象方法,抽象方法没有方法体。抽象方法用来描述系统具有什么功能,但不提供具体的实现,把具体实现留给继承该类的子类。

抽象类特点:

  • 含有抽象方法的类必须声明为抽象类(不管其中是否有其他方法);
  • 抽象类可以没有抽象方法,可以有普通方法;
  • 抽象类必须被继承,抽象方法必须被重写(若子类还是抽象类,不需要重写);
  • 抽象类不能被实例化(不能直接构造一个该类的对象)。

抽象方法特点:

  • 在类中没有方法体(抽象方法只需声明,而不需实现某些功能);
  • 抽象类中的抽象方法必须被实现;
  • 如果一个子类没有实现父类中的抽象方法,则子类也变成了一个抽象类。

接口 interface 中的方法默认为public abstract publicabstract可以省略),变量默认为public static final;类中的方法全部都是抽象方法。只有声明没有实现,在不同类中有不同的方法实现。

不同点:

  • 接口中只能包含抽象方法和默认方法,不能为普通方法提供方法实现;抽象类中可以包含普通方法;
  • 接口里不能定义静态方法(jdk1.8下可以定义static方法),抽象类可以定义静态方法;
  • 接口中只能定义静态常量,不能定义普通成员变量;抽象类即可以定义变量又可以定义静态常量;
  • 接口中不包含构造器,抽象类里可以包含构造器,抽象类中的构造器并不是用于创建对象,而是让其他子类调用这些构造器来完成抽象类的初始化操作;
  • 接口里不能包含初始化块,但抽象类可以包含;
  • 一个类最多只能有一个父类,包括抽象类;但一个类可以直接实现多个接口,通过实现多个接口可以弥补Java单继承的不足。

共同点:

  • 接口和抽象类都不能被实例化,都位于继承树的顶端,用于被其他类实现的继承;
  • 接口和抽象类都可以包含抽象方法,实现接口和继承抽象类的普通子类都必须实现这些方法。

img

final关键字
  • final修饰的,就是最终类,不能被继承。
  • final修饰的方法,就是最终方法,最终方法不能被重写
  • final修饰一个引用变量时,是指引用变量不能变,引用变量所指向的对象中的内容还是可以改变的。修饰基本数据类型变量时,内容不能变。
  • final成员变量必须在初始化代码块或在构造器中初始化。

作用:

  • final类:如果一个类不需要有子类,类的实现细节不允许改变,并且确信这个类不会再被扩展,那么就设计成final类。
  • final方法:①把方法锁定,防止任何继承类修改它的意义和实现。②高效,编译器在遇到调用final方法时候会转入内嵌机制,大大提升执行效率。
static关键字
  • static修饰的变量称为静态变量,静态变量属于整个类,而局部变量属于方法,只在该方法内有效。static不能修饰局部变量。static方法内部不能调用非静态方法。
  • 静态变量只能在类主体中定义,不能在方法中定义;
  • static变量只会创建一份,不管创建几个对象,都共用一个变量。

类方法指被static修饰的方法,无this指针。其他的就是实例方法。类方法可以调用其他类的static方法。 类方法和对象方法的区别:

1、 类方法是属于整个类的,而实例方法是属于类的某个对象的。 由于类方法是属于整个类的,并不属于类的哪个对象,所以类方法的方法体中不能有与类的对象有关的内容。即类方法体有如下限制:

  • 类方法中不能引用对象变量;
  • 类方法中不能调用类的对象方法;
  • 在类方法中不能使用superthis关键字。(this表示当前类的对象,由static修饰的方法是类直接调用,不需要创建对象,所以不能用this);
  • 类方法不能被覆盖。

2、与类方法相比,对象方法几乎没有什么限制:

  • 对象方法中可以引用对象变量,也可以引用类变量;
  • 对象方法中可以调用类方法;
  • 对象方法中可以使用superthis关键字。

static关键字的作用

  • 为某特定数据类型或对象分配单一的存储空间,而与创建对象的个数无关;实现某个方法或属性与类而不是对象关联在一起;
  • 静态变量属于类,在内存中只有一个复制,只要静态变量所在的类被加载,这个静态变量就会被分配空间。
多态
  • 定义:不同类的对象对同一消息做出响应。同一消息可以根据发送对象的不同而采用多种不同的行为方式;

  • 多态存在的三个必要条件:继承、重写、父类引用指向子类对象

  • Java中多态的实现方式:接口实现,继承父类进行方法重写,同一个类中进行方法重载;

  • 父类引用指向子类对象,该引用不能再访问子类新增的成员。Animal cat = new Cat()与直接new一个父类实例(Animal a = new Animal())的区别? 答:当父类是接口和抽象类时,不能实例化,只能运用多态,向上转型。普通类中,可以在子类中重写父类中的方法,这样就可以访问子类中的重写方法。

package case2;
import java.util.Scanner;

public class Task2 {

	public static void main(String[] args) {
		Scanner sc = new Scanner(System.in);
		String cName = sc.next();
		String cSex = sc.next();
		int cAge = sc.nextInt();
		String eName = sc.next();
		String eSex = sc.next();
		int eAge = sc.nextInt();
		// 创建测试类对象test
		// 创建Person类对象person1,引用指向中国人,通过有参构造函数实例化中国人类对象
		// 通过showEat()方法调用Chinese的eat()方法
		// 创建Person类对象person2,引用指向英国人,通过有参构造函数实例化英国人类对象
		// 通过showEat()方法调用English的eat()方法
		/********* begin *********/
		Person person1 = new Chinese(cName,cSex,cAge);	//抽象类Person不能被实例化
		showEat(person1);	//showEat是静态方法可以调用
		Person person2=new English(eName,eSex,eAge);
		showEat(person2);
		/********* end *********/
		// 强制类型转换(向下转型) 调用Chinese类特有的方法shadowBoxing()
		// 强制类型转换(向下转型) 调用English类特有的方法horseRiding()
		/********* begin *********/
		Chinese d = (Chinese) person1;
		d.shadowBoxing();	//使用特有的方法就需要向下转型,因为person1属于Person类
		English f =(English) person2;
		f.horseRiding();
		/********* end *********/
	}

	// 定义showEat方法,使用父类作为方法的形参,实现多态,传入的是哪个具体对象就调用哪个对象的eat()方法
	/********* begin *********/
	public static void showEat(Person p){
		p.eat();
	}
	/********* end *********/
}

// 抽象类Person 封装属性name、sex和age
// 构造函数初始化name、sex和age
// 声明抽象方法eat()
abstract class Person {
	/********* begin *********/
	String name;String sex;int age;
    Person(String name,String sex,int age){
        this.name=name;this.sex=sex;this.age=age;
    }
    abstract void eat();
	/********* end *********/
}

// Chinese类继承自Person类
// 构造函数初始化name、sex和age
// 重写父类方法eat() 输出'姓名:name,性别:sex,年龄:age,我是中国人,我喜欢吃饭!'
// 定义子类特有方法shadowBoxing(),当父类引用指向子类对象时无法调用该方法 输出'name在练习太极拳!'
class Chinese extends Person {
	/********* begin *********/
	public Chinese(String name,String sex,int age){
		super(name,sex,age);

	}
	/********* end *********/
	@Override
    void eat(){
        System.out.println("姓名:"+name+",性别:"+sex+",年龄:"+age+",我是中国人,我喜欢吃饭!");
    }
    void shadowBoxing(){
        System.out.println(name+"在练习太极拳!");
    }
}

// English类继承自Person类
// 构造函数初始化name、sex和age
// 重写父类方法eat() 输出'姓名:name,性别:sex,年龄:age,我是英国人,我喜欢吃三明治!'
// 定义子类特有方法horseRiding(),当父类引用指向子类对象时无法调用该方法 输出'name在练习骑马!'
class English extends Person {
	/********* begin *********/
	public English(String name,String sex,int age){
		super(name,sex,age);
	}
	@Override
	void eat(){
		System.out.println("姓名:"+name+",性别:"+sex+",年龄:"+age+",我是英国人,我喜欢吃三明治!");
	}
	void horseRiding(){
		System.out.println(name+"在练习骑马!");
	}
	/********* end *********/
}
/*知识点目录
1,Java继承
1.1 继承的概念
1.2 继承的特性
1.3 继承关键字
1.4 构造器
2,Java重写(Override)与重载(Overload)
2.1 重写(Override)
2.2 重载(Overload)
2.3 总结
3,Java多态
3.1 多态的实现方式
4,Java抽象类
5,Java封装
6,Java接口
*/
/* 1.1继承的概念
继承在本职上是特殊——一般的关系,即常说的is-a关系。子类继承父类,表明子类是一种特殊的父类,并且具有父类所不具有的一些属性或方法。通过 extends 关键字可以声明一个类是从另外一个类继承而来的。
*/
/* 1.2继承的特性
1).子类拥有父类非private的属性,方法;
2).子类可以拥有自己的属性和方法,即子类可以对父类进行扩展;
3).子类可以用自己的方式实现父类的方法;
4).Java的继承是单继承,这是Java继承区别于C++继承的一个特性;
5).提高了类之间的耦合性(继承的缺点,耦合度高就会造成代码之间的联系)。
*/
/* 1.3继承关键字
1).使用 extends 和 implements 来实现继承,所有的类都是继承于 java.lang.Object,当一个类没有继承的两个关键字,则默认继承Object;
2).一个子类只能拥有一个父类,所以 extends 只能继承一个类;
3).使用 implements 关键字变相的使Java具有多继承的特性,为类继承接口,可以同时继承多个接口;
4).通过super关键字来实现对父类成员的访问,用来引用当前对象的父类;
5).final 关键字
5.1) 声明类则把类定义为不能继承的,即最终类;修饰方法,则该方法不能被子类重写;定义实例变量,则变量不能被修改;
5.2) final 类的方法自动为 final方法,但实例变量不自动是 final变量。
*/
interface A{}
interface B{}
class Animal{
   public void move(){
       System.out.println("动物可以移动");
   }
   Animal(){
       System.out.println("Animal()");
   }
   Animal(int n){
       System.out.println("Animal(int)");
   }
}
final class Dog extends Animal implements A,B {
   final String name = "旺财";
   final public void finalFun(){
   }
   /* 2.1.1 重写(Override)
    1).重写是子类对父类的允许访问的方法的实现过程进行重新编写, 返回值和形参都不能改变。即外壳不变,核心重写;
    2).重写的好处在于子类可以根据需要,定义特定于自己的行为。也就是说子类能够根据需要实现父类的方法;
    3).重写方法不能抛出新的检查异常或者比被重写方法声明更加宽泛的异常。
    */
   public void move(){
       System.out.println("狗可以跳墙");
   }
   /*2.2.1重载(Overload)
   1).重载是在一个类里面,方法名字相同,而参数不同。返回类型可以相同也可以不同;
   2).每个重载的方法(或者构造函数)都必须有一个独一无二的参数类型列表。
    */
   public void move(int n){
       System.out.println("这只狗可以跳"+n+"米的墙");
   }
   public String toString() {
       //1.3.1 通过super关键字来实现对父类成员的访问,用来引用当前对象的父类。
       return "I am a Dog " + super.toString();
   }
/* 1.4 构造器
1).子类不能继承父类的构造器(构造方法或者构造函数),但是父类的构造器带有参数的,则必须在子类的构造器中显式地通过super关键字调用父类的构造器并配以适当的参数列表;
2).如果父类有无参构造器,则在子类的构造器中用super调用父类构造器不是必须的,如果没有使用super关键字,系统会自动调用父类的无参构造器。
*/
   Dog(){
       super(300);
       System.out.println("Dog()");
   }
   Dog(int n){
       System.out.println("Dog(int)");
   }
}
class Test{
   static void testsuper(){
       System.out.println("-----testsuper----");
       new Dog();
       new Dog(1);
   }
   static void testoverride(){
       System.out.println("-----testoverride-----");
       Animal a = new Animal();
       Animal b = new Dog();
       a.move();
       /*2.1.2  重写(Override)
       1).尽管b属于Animal类型,但是它运行的是Dog类的move方法;
       2).在编译阶段,只是检查参数的引用类型。在运行时,Java虚拟机(JVM)指定对象的类型并且运行该对象的方法。
        */
       b.move();
   }
}
public class JavaDemo{
   public static void main(String[] args) {
       Test.testsuper();
       Test.testoverride();
   }
}
/* 2.1.3方法的重写规则
1).参数列表必须完全与被重写方法的相同;
2).返回类型必须完全与被重写方法的返回类型相同;
3).访问权限不能比父类中被重写的方法的访问权限更低;
4).父类的成员方法只能被它的子类重写;
5).声明为final的方法不能被重写;
6).声明为static的方法不能被重写,但是能够被再次声明;
7).子类和父类在同一个包中,那么子类可以重写父类所有方法,除了声明为private和final的方法;
8).子类和父类不在同一个包中,那么子类只能够重写父类的声明为public和protected的非final方法;
9).重写的方法能够抛出任何非强制异常,无论被重写的方法是否抛出异常,但是,重写的方法不能抛出新的强制性异常,或者比被重写方法声明的更广泛的强制性异常,反之则可以;
10).构造方法不能被重写;
11).如果不能继承一个方法,则不能重写这个方法。
*/
/*2.2.2重载规则
1).被重载的方法必须改变参数列表(参数个数或类型或顺序不一样);
2).被重载的方法可以改变返回类型;
3).被重载的方法可以改变访问修饰符;
4).被重载的方法可以声明新的或更广的检查异常;
5).方法能够在同一个类中或者在一个子类中被重载;
6).无法以返回值类型作为重载函数的区分标准。
*/
/*
重写与重载之间的区别
区别点            重载方法        重写方法
参数列表       必须修改        一定不能修改
返回类型       可以修改        一定不能修改
异常                可以修改        可以减少或删除,一定不能抛出新的或者更广的异常
访问                可以修改        一定不能做更严格的限制(可以降低限制)
总结
方法的重写和重载是Java多态性的不同表现,重写是父类与子类之间多态性的一种表现,重载是一类中多态性的一种表现。
*/
/*访问控制修饰符
Java中,可以使用访问控制符来保护对类、变量、方法和构造方法的访问。Java支持4种不同的访问权限。
默认的,也称为 default,在同一包内可见,不使用任何修饰符;
私有的,以 private 修饰符指定,在同一类内可见;
共有的,以 public 修饰符指定,对所有类可见;
受保护的,以 protected 修饰符指定,对同一包内的类和所有子类可见。
我们可以可以通过以下表来说明访问权限:
访问控制
修饰符      当前类      同一包内     子孙类      其他包
public             Y                   Y                  Y                   Y
protected          Y                   Y                  Y                   N
default            Y                   Y                  N                   N
private            Y                   N                  N                   N
*/
/*3,Java多态
多态就是同一个接口,使用不同的实例执行不同操作。
3.1 多态的实现方式 重写、接口、抽象类和抽象方法。
*/
/*4,Java抽象类
1).使用abstract class来定义抽象类,抽象类不能被实例化;
2).抽象类中不一定包含抽象方法,但是有抽象方法的类必定是抽象类;
3).抽象类中的抽象方法只是声明,不包含方法体;
4).构造方法,类方法(用static修饰的方法)不能声明为抽象方法;
5).任何子类必须重写父类的抽象方法,或者声明自身为抽象类。
*/
abstract class Employee{
   //抽象方法
   public abstract double computePay();
}
/*4.2继承抽象类
*/
class Salary extends Employee{
   public double computePay(){ 
       return 0;   
   }
}
/*
5,Java封装
封装(英语:Encapsulation)是指,一种将抽象性函式接口的实现细节部份包装、隐藏起来的方法。
封装的优点:
1).良好的封装能够减少耦合;
2).类内部的结构可以自由修改;
3). 可以对成员变量进行更精确的控制;
4). 隐藏信息,实现细节。
实现Java封装的步骤:
1). 修改属性的可见性来限制对属性的访问;
2).对每个值属性提供对外的公共方法访问。
*/
/*
6,Java接口
1).接口,在Java编程语言中是一个抽象类型,是抽象方法的集合,接口通常以interface来声明;
2).一个实现接口的类,必须实现接口内所描述的所有方法,否则就必须声明为抽象类;
3).接口不能包含成员变量,除了 static 和 final 变量;
4).接口中每一个方法也是隐式抽象的,接口中的方法会被隐式的指定为 public abstract;
5).接口中可以含有变量,但是接口中的变量会被隐式的指定为 public static final 变量;
6).接口中的方法是不能在接口中实现的,只能由实现接口的类来实现接口中的方法;
7).类在实现接口的方法时,不能抛出强制性异常,只能在接口中,或者继承接口的抽象类中抛出该强制性异常;
8).类在重写方法时要保持一致的方法名,并且应该保持相同或者相兼容的返回值类型;
9).接口的继承使用extends关键字,允许多重继承,可能定义或继承相同的方法。
*/
interface C extends A,B{}

package case3;


import java.util.Scanner;

public class Task3 {
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        String pppName = sc.next();
        int pppAge = sc.nextInt();
        String bpName = sc.next();
        int bpAge = sc.nextInt();
        String ppcName = sc.next();
        int ppcAge = sc.nextInt();
        String bcName = sc.next();
        int bcAge = sc.nextInt();
        // 测试运动员(乒乓球运动员和篮球运动员)
        // 乒乓球运动员
        // 通过带参构造函数实例化PingPangPlayer对象ppp
        // 输出'name---age'
        // 分别调用sleep()、eat()、study()、speak()方法
        /********* begin *********/
        PingPangPlayer ppp = new PingPangPlayer(pppName, pppAge);
        System.out.println(pppName+"---"+pppAge);
        ppp.sleep();ppp.eat();ppp.study();ppp.speak();
        /********* end *********/
        System.out.println("----------------");
        // 篮球运动员
        // 通过带参构造函数实例化BasketballPlayer对象bp
        // 输出'name---age'
        // 分别调用sleep()、eat()、study()方法
        /********* begin *********/
        BasketballPlayer bp = new BasketballPlayer(bpName, bpAge);
        System.out.println(bpName+"---"+bpAge);
        bp.sleep();bp.eat();bp.study();
        /********* end *********/
        System.out.println("----------------");
        // 测试教练(乒乓球教练和篮球教练)
        // 乒乓球教练
        // 通过带参构造函数实例化PingPangCoach对象ppc
        // 输出'name---age'
        // 分别调用sleep()、eat()、teach()、speak()方法
        /********* begin *********/
        PingPangCoach ppc = new PingPangCoach(ppcName,ppcAge);
        System.out.println(ppcName+"---"+ppcAge);
        ppc.sleep();ppc.eat();ppc.teach();ppc.speak();
        /********* end *********/
        System.out.println("----------------");
        // 篮球教练
        // 通过带参构造函数实例化BasketballCoach对象bc
        // 输出'name---age'
        // 分别调用sleep()、eat()、teach()方法
        /********* begin *********/
        BasketballCoach bc = new BasketballCoach(bcName,bcAge);
        System.out.println(bcName+"---"+bcAge);
        bc.sleep();bc.eat();bc.teach();
        /********* end *********/
        System.out.println("----------------");
    }
}

// 说英语接口 声明抽象方法speak()
interface SpeakEnglish {
    /********* begin *********/
    abstract void speak();
    /********* end *********/
}

// 定义人的抽象类Person 封装name和age
// 无参构造函数
// 有参构造函数初始化name和age
// 定义具体方法sleep() 输出'人都是要睡觉的'
// 抽象方法eat()(吃的不一样)
abstract class Person {
    /********* begin *********/
    private String name;
    private int age;
    public Person(){}
    public Person(String name,int age){
        this.name=name;
        this.age=age;
    }
    public void sleep(){
        System.out.println("人都是要睡觉的");
    }
    public String getName(){return name;}
    public int getAge(){return age;}
    /********* end *********/
}

// 定义运动员Player(抽象类)继承自Person类
// 无参构造函数
// 有参构造函数初始化name和age
// 运动员学习内容不一样,抽取为抽象 定义抽象方法study()
abstract class Player extends Person {
    private String name;
    private int age;
    /********* begin *********/
    public Player(){}
    public Player(String name,int age){
        super(name,age);
    }
    abstract void study();
    public String getName(){return name;}
    public int getAge(){return age;}
    /********* end *********/
}

// 定义教练Coach(抽象类)继承自Person类
// 无参构造函数
// 有参构造函数初始化name和age
// 教练教的不一样 定义抽象方法teach()
abstract class Coach extends Person {
    /********* begin *********/
    private String name;
    private int age;
    /********* begin *********/
    public Coach(){}
    public Coach(String name,int age){
        super(name,age);
    }
    abstract void teach();
    public String getName(){return name;}
    public int getAge(){return age;}
    /********* end *********/
}

// 定义乒乓球运动员具体类PingPangPlayer 继承自Player类并实现SpeakEnglish类(兵乓球运动员需要说英语)
// 无参构造函数
// 有参构造函数初始化name和age
// 实现自己的eat()方法 输出'乒乓球运动员吃大白菜,喝小米粥'
// 实现自己的study()方法 输出'乒乓球运动员学习如何发球和接球'
// 实现自己的speak()方法 输出'乒乓球运动员说英语'
class PingPangPlayer extends Player implements SpeakEnglish {
    /********* begin *********/
    String name;int age;
    public PingPangPlayer(){}
    public PingPangPlayer(String name,int age){
        super(name,age);
    }
    public void eat(){
        System.out.println("乒乓球运动员吃大白菜,喝小米粥");
    }
    public void study(){
        System.out.println("乒乓球运动员学习如何发球和接球");
    }
    public void speak(){
        System.out.println("乒乓球运员说英语");
    }
    /********* end *********/
}

// 定义篮球运动员具体类BasketballPlayer 继承自Player类 不需要继承接口,因为他不需要说英语
// 无参构造函数
// 有参构造函数初始化name和age
// 实现自己的eat()方法 输出'篮球运动员吃牛肉,喝牛奶'
// 实现自己的study()方法 输出'篮球运动员学习如何运球和投篮'
class BasketballPlayer extends Player {
    /********* begin *********/
    String name;int age;
    public BasketballPlayer(){}
    public BasketballPlayer(String name,int age){
        super(name,age);
    }
    public void eat(){
        System.out.println("篮球运动员吃牛肉,喝牛奶");
    }
    public void study(){
        System.out.println("篮球运动员学习如何运球和投篮");
    }
    /********* end *********/
}

// 定义乒乓球教练具体类 PingPangCoach 继承自Coach类并实现SpeakEnglish类(兵乓球教练需要说英语)
// 无参构造函数
// 有参构造函数初始化name和age
// 实现自己的eat()方法 输出'乒乓球教练吃小白菜,喝大米粥'
// 实现自己的teach()方法 输出'乒乓球教练教如何发球和接球'
// 实现自己的speak()方法 输出'乒乓球教练说英语'
class PingPangCoach extends Coach implements SpeakEnglish {
    /********* begin *********/
    String name;int age;
    public PingPangCoach(){}
    public PingPangCoach(String name,int age){
        super(name,age);
    }
    public void eat(){
        System.out.println("乒乓球教练吃小白菜,喝大米粥");
    }
    public void teach(){
        System.out.println("乒乓球教练教如何发球和接球");
    }
    public void speak(){
        System.out.println("乒乓球教练说英语");
    }
    /********* end *********/
}

// 定义篮球教练具体类BasketballCoach 继承自Coach类 不需要继承接口,因为他不需要说英语
// 无参构造函数
// 有参构造函数初始化name和age
// 实现自己的eat()方法 输出'篮球教练吃羊肉,喝羊奶'
// 实现自己的teach()方法 输出'篮球教练教如何运球和投篮'
class BasketballCoach extends Coach {
    /********* begin *********/
    String name;int age;
    public BasketballCoach(){}
    public BasketballCoach(String name,int age){
        super(name,age);
    }
    public void eat(){
        System.out.println("篮球教练吃羊肉,喝羊奶");
    }
    public void teach(){
        System.out.println("篮球教练教如何运球和投篮");
    }
    /********* end *********/
}

推荐阅读