目录
一、继承性
继承的语法规则
继承性的体现
继承性的作用
java中继承的规则
练习
二、方法的重写(Override/ Overwrite)
定义
要求
区分方法的重写与重载
方法的重写
方法的重载
练习
三、关键字super
super的使用
super使用的注意事项
this和super的区别
练习
“继承”是面向对象软件技术当中的一个概念。如果多个类存在相同的属性和行为,将这些内容抽取出来单独存放在一个类中,那么多个类就无法在定义这些属性和行为,只要继承那个类即可。
此处的多个类称为子类(派生类),单独的类称为父类(基类)。
class Subclass extends SuperClass{}
1、一旦子类A继承了父类B,那个子类A中就获取了父类B中声明的属性和方法
特别的:父类中声明为私有的属性和方法,子类继承父类以后,依然认为获取了父类中的私有的结构,只是因为封装性的影响使得子类无法直接调用而已。
封装性解决可见性问题,而继承性解决子类是否能拿到父类中的结构(属性、方法)。
2、子类继承父类以后可以直接使用父类中的属性及方法,还可以定义自身特有的属性和方法,实现功能的拓展。
1、减少了代码冗余,提高了代码的可复用性
2、便于功能的拓展。继承的关键字是“extends”,英文含义是扩展、延展,可以理解为是对父类功能的一个拓展。
3、继承的出现使类与类之间产生了关系,为之后的多态性使用提供了前提
1、子类不能直接访问父类中私有的属性及方法
2、java只支持单继承和多层继承,不支持多重继承。 即一个类只能有一个父类;而一个父类可以派生出多个子类。
3、子类的直接继承的父类称为直接父类;间接继承的父类称为间接父类
4、如果一个类没有显式的声明继承于某一个类的话,那么此类默认继承于java.lang.Object类。即所有类都能使用java.lang.Object类中生命的功能。
1、. (1)定义一个ManKind类,包括 :
成员变量int sex和int salary;
方法void manOrWoman():根据sex的值显示“man”(sex==1)或者“woman”(sex==0);
方法void employeed():根据salary的值显示“no job”(salary==0)或者“ job”(salary!=0)。
(2)定义类Kids继承ManKind,并包括:
成员变量int yearsOld;
方法printAge()打印yearsOld的值。
(3)定义类KidsTest,在类的main方法中实例化Kids的对象someKid,用该对象访问其父类的成员变量及方法。
public class ExtendsTest001 {
public static void main(String[] args) {
Kids someKid=new Kids();
someKid.setSex(1);
someKid.setYearsOld(18);
someKid.manOrWoman();
someKid.printAge();
someKid.setSalary(0);
}
}
class Mankind{
private int sex;
private int salary;
public int getSex() {
return sex;
}
public void setSex(int sex) {
this.sex = sex;
}
public int getSalary() {
return salary;
}
public void setSalary(int salary) {
this.salary = salary;
}
public void manOrWoman() {
if(this.sex==1) {
System.out.println("我是 man");
}
if(this.sex==0) {
System.out.println("我是 woman");
}
}
public void employed() {
if(this.salary==0) {
System.out.println("I hava no job");
}else {
System.out.println("I have a job");
}
}
}
class Kids extends Mankind{
private int yearsOld;
public int getYearsOld() {
return yearsOld;
}
public void setYearsOld(int yearsOld) {
this.yearsOld = yearsOld;
}
public void printAge(){
System.out.println("我今年"+yearsOld+"岁了");
}
}
运行结果如下:
2、. 根据下图实现类。在CylinderTest类中创建Cylinder类的对象,设置圆 柱的底面半径和高,并输出圆柱的体积。
public class CylinderTest {
public static void main(String[] args) {
Cylinder cylinder=new Cylinder();
cylinder.setRadius(2);
cylinder.setLength(4);
double volume=cylinder.findVolume();
System.out.println("Volum is "+volume);
}
}
class Circle{
private double radius;
public Circle(){
radius=1;
}
public double getRadius() {
return radius;
}
public void setRadius(double radius) {
this.radius = radius;
}
public double findArea() {
return Math.PI*Math.pow(getRadius(), 2);
}
}
class Cylinder extends Circle{
private double length;
public Cylinder() {
length=1;
}
public double getLength() {
return length;
}
public void setLength(double length) {
this.length = length;
}
public double findVolume() {
return findArea()*getLength();
}
}
运行结果如下:
在子类中可以对从父类中继承过来的方法进行改造,也称为方法的重置、覆盖。在程序执行时,子类方法将覆盖父类的方法。
1、子类重写的方法必须和父类被重写的方法具有相同的方法名称、参数列表。父类方法被重写后,当创建子类对象去调用父类同名同参数的方法时,实际调用的是子类重写父类的方法
2、子类重写的方法返回值类型不能大于父类被重写的方法的返回值类型
3、子类重写的方法使用的访问权限不能小于父类被重写的方法的访问权限
4、子类不能重写父类中声明为private权限的方法
5、子类方法抛出的异常不能大于父类被重写方法的异常
6、子类和父类的同名的参数的方法要么都声明为非static的,要么就都声明为static的。
说明:static类的方法一定不可以被重写
方法的重写规定方法的声明为:
权限修饰符 返回值类型 方法名 (形参列表) throws 异常类型{
//方法体;
}
1、子类重写方法的方法名的参数列表与父类中被重写分方法的的方法名和参数列表相同
2、子类重写的方法权限修饰符要不小于父类中被重写的方法的权限修饰符
3、子类不能重写父类中声明为private权限的方法
4、子类中重写方法的返回值类型要不大于父类中被重写方法的返回值类型
5、子类重写的方法抛出的异常不大于父类被重写的方法抛出的异常
1、在一个类中,允许存在一个以上的同名方法,只要他们的参数个数或者参数类型不同即可。
2、方法的重载与返回值类型无关,只看参数列表,且参数列表必须不同。调用时,根据参数列表的不同来区别。
特别的:重载实现于一个类中;重写实现于子类中。且方法的重载不表现为多态性,而方法的重写表现为多态性
修改继承性练习1中定义的类Kids,在Kids中重新定义employeed()方 法,覆盖父类ManKind中定义的employeed()方法,输出“Kids should study and no job.”
public class OverrideTest001 {
public static void main(String[] args) {
Kids someKid=new Kids();
someKid.setSex(1);
someKid.setYearsOld(18);
someKid.manOrWoman();
someKid.printAge();
someKid.setSalary(0);
someKid.employed();
}
}
class Mankind{
private int sex;
private int salary;
public int getSex() {
return sex;
}
public void setSex(int sex) {
this.sex = sex;
}
public int getSalary() {
return salary;
}
public void setSalary(int salary) {
this.salary = salary;
}
public void manOrWoman() {
if(this.sex==1) {
System.out.println("我是 man");
}
if(this.sex==0) {
System.out.println("我是 woman");
}
}
public void employed() {
if(this.salary==0) {
System.out.println("I hava no job");
}else {
System.out.println("I have a job");
}
}
}
class Kids extends Mankind{
private int yearsOld;
public int getYearsOld() {
return yearsOld;
}
public void setYearsOld(int yearsOld) {
this.yearsOld = yearsOld;
}
public void printAge(){
System.out.println("我今年"+yearsOld+"岁了");
}
@Override
public void employed() {
System.out.println("childern should study and no job");
}
}
运行结果如下:
super可以理解为“父类的···”
1、super可以用来调用父类中定义的属性
2、super可以用来调用父类中定义的方法
说明:
3、super可以用于子类构造器中用来调用父类的构造器
说明:
1、super的追溯不仅限于直接父类
2、super和this的用法相像,this代表对本类对象的引用,super代表父类的内存空间标识
3、当父类中没有空参数的构造器时,子类的构造器必须通过this(参数列表)或者super(参数列表)语句指定调用本类或父类中相应的构造器,只能“二选一”,且必须位于构造器首行
4、如果子类中的构造器中没有显式的调用父类或本类的构造器,且父类中又没有无参的构造器,则编译报错
区别点 | this | super |
---|---|---|
访问属性 | 访问本类中的属性,如果本类中没有此属性则会从父类中继续查找 | 直接访问父类中的属性 |
访问方法 | 访问本类中的方法,如果本类中没有此方法则从父类中继续查找 | 直接访问父类中的方法 |
调用构造器 | 调用本类构造器,必须放在构造器的首行 | 调用父类构造器,必须放在子类构造器首行 |
(2条消息) Java面向对象实验三_要向着光的博客-CSDN博客https://blog.csdn.net/zssxcj/article/details/127531406?spm=1001.2014.3001.5502