java公有继承的特点_Java 面向对象(十)面向对象特征之二:继承之后特点

一、继承后的特点——成员变量

当类之间产生了关系后,其中各类中的成员变量会产生影响,分为下面两类来讨论:

1、成员变量不重名

如果子类父类中出现不重名的成员变量,这时候的访问是没有影响的。

Demo:

1 //父类

2 classFu {3 //Fu中的成员变量。

4 int num = 5;5}6 //子类

7 class Zi extendsFu {8 //Zi中的成员变量

9 int num2 = 6;10 //Zi中的成员方法

11 public voidshow() {12 //访问父类中的num,

13 System.out.println("Fu num="+num); //本类中没有,继承而来,所以直接访问。

14 //访问子类中的num2

15 System.out.println("Zi num2="+num2); //本类中有,访问本类中num2

16}17}18 //测试类

19 classTest{20 public static voidmain(String[] args) {21 //创建子类对象

22 Zi z = newZi();23 //调用子类中的show方法

24z.show();25}26}27

28演示结果:29 Fu num = 5

30 Zi num2 = 6

2、成员变量重名

如果子类父类中出现了重名的成员变量,这时候的访问是有影响的。

Demo:

1 1 classFu {2 2 //Fu中的成员变量。

3 3 int num = 5;4 4}5 5 class Zi extendsFu {6 6 //Zi中的成员变量

7 7 int num = 6;8 8 public voidshow() {9 9 //访问父类中的num

10 10 System.out.println("Fu num=" +num);11 11 //访问子类中的num

12 12 System.out.println("Zi num=" +num);13 13}14 14}15 15 classExtendsDemo03 {16 16 public static voidmain(String[] args) {17 17 //创建子类对象

18 18 Zi z = newZi();19 19 //调用子类中的show方法

20 20z.show();21 21}22 22}23 23演示结果:24 24 Fu num = 6

25 25 Zi num = 6

通过以上Demo可以看出,子父类中出现了同名的成员变量时,在子类中访问父类是访问不到的。

这时需要另外一个关键字——super 关键字。

在子类中需要访问父类中非私有成员变量时,需要使用 super 关键字,修饰父类成员变量,类似于之前的 this。

使用格式:

super.父类成员变量名

将上面的 Demo 中子类方法进行修改:

1 class Zi extendsFu {2   //Zi中的成员变量

3   int num = 6;4   public voidshow() {5     //访问父类中的num

6     System.out.println("Fu num=" + super.num);7     //访问子类中的num

8     System.out.println("Zi num=" + this.num);9} 10} 11演示结果: 12 Fu num = 5

13 Zi num = 6

Tips:父类中的成员变量是非私有的,子类中可以直接访问。若父类中的成员变量私有了,子类是不能直接访问的。通常,遵循封装的原则,使用 private 修饰成员变量 ,那么访问父类的成员变量就需要使用公开的setter 和 getter 方法。

二、继承后的特点——成员方法

当类之间产生了关系,其中各类中的成员方法又会产生影响,下面分两种情况来讨论:

1、成员方法不重名

如果子类父类中出现不重名的成员方法,这时调用是没有影响的。对象调用方法时,会先在子类中查找有没有对应的方法,若子类中存在就会执行子类中的方法,若子类中不存在就会执行父类(直接父类与间接父类,会一直向上找)中相应的方法。

Demo:

1 classFu{2   public voidshow(){3     System.out.println("Fu类中的show方法执行");4}5}6 class Zi extendsFu{7   public voidshow2(){8     System.out.println("Zi类中的show2方法执行");9}10}11 public classExtendsDemo04{12   public static voidmain(String[] args) {13     Zi z = newZi();14     //子类中没有show方法,但是可以找到父类方法去执行

15z.show();16z.show2();17}18 }

2、成员方法重名——重写(Override/overwrite)

如果子类父类中出现 重名 的成员方法,这时的访问是一种特殊情况,叫做方法重写(Override)。

方法重写:子类中出现与父类一模一样的方法时(返回值类型,方法名和参数列表都相同),会出现覆盖效果,也称为重写或者复写。声明不变,重新实现。

Demo:

1 classFu {2   public voidshow() {3     System.out.println("Fu show");4}5}6 class Zi extendsFu {7   //子类重写了父类的show方法

8   public voidshow() {9     System.out.println("Zi show");10}11}12 public classExtendsDemo05{13   public static voidmain(String[] args) {14     Zi z = newZi();15     //子类中有show方法,只执行重写后的show方法

16     z.show(); //Zi show

17}18 }

3、方法的重写(详解)

(1)定义:在子类中可以根据需要对从父类中继承来的方法进行改造, 也称为方法的重置、覆盖。在程序执行时,子类的方法将覆盖父类的方法。

(2)重写的规定:

方法的声明: 权限修饰符 返回值类型 方法名(形参列表) throws 异常的类型{

//方法体

}

约定俗称:子类中的叫重写的方法,父类中的叫被重写的方法;

(3)重写的要点

① 子类重写的方法的方法名和形参列表必须与父类被重写的方法的方法名和形参列表相同;

② 权限修饰符:子类重写的方法的权限修饰符不能小于父类被重写的方法的权限修饰符;(public > protected > (default) > private)

特殊情况:子类不能重写父类中声明为private权限的方法;

③ 返回值类型:子类重写的方法的返回值类型不能大于父类被重写的方法的返回值类型

A、父类被重写的方法的返回值类型是void,则子类重写的方法的返回值类型只能是void;

B、父类被重写的方法的返回值类型是A类型,则子类重写的方法的返回值类型可以是A类或A类的子类;

C、父类被重写的方法的返回值类型是基本数据类型(比如:double),则子类重写的方法的返回值类型必须是相同的基本数据类型(必须也是double)

④ 异常类型:子类重写的方法抛出的异常类型不能大于父类被重写的方法抛出的异常类型

父类方法没有抛出异常,子类重写父类该方法时也不可抛出异常。此时子类产生该异常,只能捕获处理,不能声明抛出。

注意:子类与父类中同名同参数的方法必须同时声明为非static的(即为重写),或者同时声明为static的(不是重写) 。因为static方法是属于类的,子类无法覆盖父类的方法。

4、重写的应用

重写以后,当创建子类对象以后,通过子类对象调用子父类中的同名同参数的方法时,实际执行的是子类重写父类的方法。

子类可以根据需要,定义特定与自己的行为。既沿袭了父类的功能名称,又根据子类的需要重新实现父类方法,从而进行扩展增强。

Demo:新手机增加来点显示头像的功能。

1 classPhone {2 public voidsendMessage(){3 System.out.println("发短信");4 }5 public voidcall(){6 System.out.println("打电话");7 }8 public voidshowNum(){9 System.out.println("来电显示号码");10 }11 }12 //智能手机类

13 class NewPhone extendsPhone {14 //重写父类的来电显示号码功能,并增加自己的显示姓名和图片功能

15 public voidshowNum(){16 //调用父类已经存在的功能使用super

17 super.showNum();18 //增加自己特有显示姓名和图片功能

19 System.out.println("显示来电姓名");20 System.out.println("显示头像");21 }22 }23

24 public classExtendsDemo06 {25 public static voidmain(String[] args) {26 //创建子类对象

27 NewPhone np = newNewPhone();28 //调用父类继承而来的方法

29 np.call();30 //调用子类重写的方法

31 np.showNum();32 }33 }

Tips:这里进行重写时,用到 super.父类成员方法,表示调用父类的成员方法。

三、继承后的特点——构造方法

当类之间产生了关系,现在来讨论各类中的构造方法的影响。

构造方法的定义格式和作用:

1、构造方法的名字和类名一致的,所以子类是无法继承父类构造方法的。

2、构造方法的作用是初始化成员变量的。所以子类的初始化过程中,必须先执行父类的初始化工作,父类成员变量初始化后,才可以给子类使用。

3、子类的构造方法中默认有一个super(),默认情况下,表示调用父类的无参构造方法;如果父类没有无参构造,那么在子类的构造方法的首行,必须手动调用父类的有参构造。

Demo:

1 classFu {2 private intn;3Fu(){4 System.out.println("Fu()");5}6}7 class Zi extendsFu {8Zi(){9 //super(),调用父类构造方法,默认就给提供一个。

10 super();11 System.out.println("Zi()");12}13}14 public classExtendsDemo07{15 public static voidmain (String args[]){16 Zi zi = newZi();17}18}19输出结果:20Fu()21 Zi()

小结:

子类构造方法当中有一个默认隐含的 “super()” 调用,所以一定是先调用的父类构造,后执行的子类构造;

子类构造可以通过super关键字来调用父类重载构造;

super的父类构造调用,必须是子类构造方法的第一个语句。不能一个子类构造调用多次super构造,只能有一个;

子类中所有的构造器默认都会访问父类中空参数的构造器;

当父类中没有空参数的构造器时, 子类的构造器必须通过 this(参数列表)或者super(参数列表)语句指定调用本类或者父类中相应的造器。 同时, 只能”二选一”, 且必须放在构造器的首行;

如果子类构造器中既未显式调用父类或本类的构造器, 且父类中又没有无参的构造器, 则编译出错。

四、super 关键字

1、super 含义

在Java类中使用 super来调用父类中的指定操作,表示从父类中去查找,引用父类的属性或方法。

注意:super 只能访问在父类中可见的属性、方法、构造器(非 private修饰)

2、用法

(1)super.属性

当子类声明了和父类同名的属性时,可以使用 super.属性来访问父类中定义属性;

(2)super.方法

当在子类中需要调用父类被重写的方法时,可以使用 super.方法,调用父类中定义的成员方法;

(3)super() 或 super(实参列表)

super():调用父类的无参构造;

super(实参列表): 调用父类的有参构造;

用于在子类构造器中调用父类的构造器。

注意:

① super() 或 super(实参列表) 必须在子类构造器的首行;

② 如果子类的构造器中,没有写 super(),它也存在;但是如果子类构造器中写super(实参列表),那么super()就不会存在的;

③ super 的追溯不仅仅限制于直接父类,也可以从间接父类中获取;

④ super 和 this 的用法想象,this 代表本类对象的应用, super 代表父类的内存空间的标识;

五、super 与 this 区别

1、父类空间优先于子类对象产生

在每次创建子类对象时,先初始化父类空间,再创建其子类对象本身。

目的在于子类对象中包含了其对应的父类空间,便可以包含其父类的成员,如果父类成员非private修饰,则子类可以随意使用父类成员。

代码体现在子类的构造方法调用时,一定先调用父类的构造方法。

图示:

28d87626b488a314e8d8eee8831cd508.png

3194ac6a02303c2006a675b57aa46fb4.png

2、super 与 this 的含义

(1)super:代表父类的存储空间标识(可以理解为父类的引用);

(2)this:代表当前对象的引用(谁调用就代表谁);

3、super 和 this 的用法

a、访问成员

this.成员变量 ‐‐ 本类的

super.成员变量 ‐‐ 父类的

this.成员方法名() ‐‐ 本类的

super.成员方法名() ‐‐ 父类的

Demo:

1 classAnimal {2 public voideat() {3 System.out.println("animal : eat");4 }5 }6

7 class Cat extendsAnimal {8 public voideat() {9 System.out.println("cat : eat");10 }11 public voideatTest() {12 this.eat(); //this 调用本类的方法

13 super.eat(); //super 调用父类的方法

14 }15 }16 public classExtendsDemo08 {17 public static voidmain(String[] args) {18 Animal a = newAnimal();19 a.eat();20 Cat c = newCat();21 c.eatTest();22 }23 }24 输出结果为:25 animal : eat26 cat : eat27 animal : eat

b、访问构造方法

this(...) ‐‐ 本类的构造方法

super(...) ‐‐ 父类的构造方法

注意:

(1)子类的每个构造方法中均有默认的 super(),调用父类的空参构造。手动调用父类构造会覆盖默认的super()。 ;

(2)super() 和 this() 都必须是在构造方法的第一行,所以不能同时出现。

4、super 与 this 的区别

c2d3b8222adf9d08ac645f3bcce30a91.png

六、继承的特点

1、子类继承了父类,子类不能直接访问父类中私有的属性、方法。可以通过 setter/getter 方法来调用。

2、子类继承父类时,构造器是不能被继承的。

3、子类继承父类时,在子类的构造器中一定要去调用父类的构造器。

默认情况下,调用的是父类的无参构造;如果父类没有无参构造,那么在子类的构造器的首行,必须手动调用父类的有参构造。

4、Java 只支持单继承,即一个 Java 类只能有一个直接父类。

Demo:

//一个类只能有一个父类,不可以有多个父类。

class C extends A{} //ok

class C extends A,B... //error

5、Java 支持多层继承(继承体系)。

Demo:

class A{}

class B extends A{}

class C extends B{}

扩展:顶层父类是 Object 类,所有的类默认继承 Object,作为父类。

6、子类和父类是一种相对的概念。

Demo:

08064fb6d612c8de9163738abaeda059.png

7、

七、子类对象实例化过程

1、子类实例化过程

d4acaacda70c9aaf2fe1b5e659e79806.png

2、案例

334ce97d6d206f0decf17681069724d7.png    

9446ddf96659b75d7fb9a0b826d197bd.png

Demo:

1 classCreature {2 publicCreature() {3 System.out.println("Creature无参数的构造器");4 }5 }6

7 class Animal extendsCreature {8 publicAnimal(String name) {9 System.out.println("Animal带一个参数的构造器,该动物的name为" +name);10 }11

12 public Animal(String name, intage) {13 this(name);14 System.out.println("Animal带两个参数的构造器,其age为" +age);15 }16 }17

18 public class Wolf extendsAnimal {19 publicWolf() {20 super("灰太狼", 3);21 System.out.println("Wolf无参数的构造器");22 }23

24 public static voidmain(String[] args) {25 newWolf();26 }27 }

3、小结

(1)从结果上来看:(继承性)

子类继承父类以后,就获取了父类中声明的属性或方法。

创建子类的对象,在堆空间,就会加载所有父类中声明的属性。

(2)从过程上来看:

当我们通过子类的构造器创建子类对象时,我们一定会直接或间接的调用其父类的构造器,进而调用父类的父类的构造器。

直到调用了java.lang.Object类中空参的构造器为止。正因为加载过所有的父类的结构,所以才可以看到内存中有父类中的结构,子类对象才可以考虑进行调用。

注意:虽然创建子类对象时,调用了父类的构造器,但是自始至终就创建过一个对象,即为new的子类对象。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值