Java类的继承实例详解(动力节点Java学院整理)

2025-05-29 0 72

一.你了解吗?

  在java中,文件是以.java为后缀的代码文件,在每个文件中最多只允许出现一个public,当有public的时候,文件的名称必须和public的名称相同,若不存在public,则文件的名称可以为任意的名称(当然以数字开头的名称是不允许的)。

  在内部,对于成员变量,如果在定义的时候没有进行显示的赋值初始化,则java会保证的每个成员变量都得到恰当的初始化:

  1)对于 char、short、byte、int、long、float、double等基本数据型的变量来说会默认初始化为0(boolean变量默认会被初始化为false);

  2)对于引用型的变量,会默认初始化为null。

  如果没有显示地定义构造器,则编译器会自动创建一个无参构造器,但是要记住一点,如果显示地定义了构造器,编译器就不会自动添加构造器。注意,所有的构造器默认为static的。

  下面我们着重讲解一下 初始化 顺序:

  当程序执行时,需要生成某个的对象,java执行引擎会先检查是否加载了这个,如果没有加载,则先执行的加载再生成对象,如果已经加载,则直接生成对象。

  在的加载过程中,的static成员变量会被初始化,另外,如果中有static语句块,则会执行static语句块。static成员变量和static语句块的执行顺序同代码中的顺序一致。记住,在java中,是按需加载,只有当需要用到这个的时候,才会加载这个,并且只会加载一次。看下面这个例子就明白了:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14
public class test {

public static void main(string[] args) throws classnotfoundexception {

bread bread1 = new bread();

bread bread2 = new bread();

}

}

class bread {

static{

system.out.println("bread is loaded");

}

public bread() {

system.out.println("bread");

}

}

  运行这段代码就会发现"bread is loaded"只会被打印一次。

  在生成对象的过程中,会先初始化对象的成员变量,然后再执行构造器。也就是说中的变量会在任何方法(包括构造器)调用之前得到初始化,即使变量散步于方法定义之间。

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16
public class test {

public static void main(string[] args) {

new meal();

}

}

class meal {

public meal() {

system.out.println("meal");

}

bread bread = new bread();

}

class bread {

public bread() {

system.out.println("bread");

}

}

  输出结果为:

bread
meal

二.你了解继承吗?

  继承是所有oop语言不可缺少的部分,在java中使用extends关键字来表示继承关系。当创建一个时,总是在继承,如果没有明确指出要继承的,就总是隐式地从根object进行继承。比如下面这段代码:

?

1

2

3

4

5

6

7

8
class person {

public person() {

}

}

class man extends person {

public man() {

}

}

  man继承于person,这样一来的话,person称为父(基),man称为子(导出)。如果两个存在继承关系,则子会自动继承父的方法和变量,在子中可以调用父的方法和变量。在java中,只允许单继承,也就是说 一个最多只能显示地继承于一个父。但是一个却可以被多个类继承,也就是说一个可以拥有多个子

  1.子继承父的成员变量

  当子继承了某个之后,便可以使用父中的成员变量,但是并不是完全继承父的所有成员变量。具体的原则如下:

  1)能够继承父的public和protected成员变量;不能够继承父的private成员变量;

  2)对于父的包访问权限成员变量,如果子和父在同一个包下,则子能够继承;否则,子不能够继承;

  3)对于子可以继承的父成员变量,如果在子中出现了同名称的成员变量,则会发生隐藏现象,即子的成员变量会屏蔽掉父的同名成员变量。如果要在子中访问父中同名成员变量,需要使用super关键字来进行引用。

  2.子继承父的方法

  同样地,子也并不是完全继承父的所有方法。

  1)能够继承父的public和protected成员方法;不能够继承父的private成员方法;

  2)对于父的包访问权限成员方法,如果子和父在同一个包下,则子能够继承;否则,子不能够继承;

  3)对于子可以继承的父成员方法,如果在子中出现了同名称的成员方法,则称为覆盖,即子的成员方法会覆盖掉父的同名成员方法。如果要在子中访问父中同名成员方法,需要使用super关键字来进行引用。

  注意:隐藏和覆盖是不同的。隐藏是针对成员变量和静态方法的,而覆盖是针对普通方法的。(后面会讲到)

  3.构造器

  子是不能够继承父的构造器,但是要注意的是,如果父的构造器都是带有参数的,则必须在子的构造器中显示地通过super关键字调用父的构造器并配以适当的参数列表。如果父有无参构造器,则在子的构造器中用super关键字调用父构造器不是必须的,如果没有使用super关键字,系统会自动调用父的无参构造器。看下面这个例子就清楚了:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22
class shape {

protected string name;

public shape(){

name = "shape";

}

public shape(string name) {

this.name = name;

}

}

class circle extends shape {

private double radius;

public circle() {

radius = 0;

}

public circle(double radius) {

this.radius = radius;

}

public circle(double radius,string name) {

this.radius = radius;

this.name = name;

}

}

  这样的代码是没有问题的,如果把父的无参构造器去掉,则下面的代码必然会出错:

Java类的继承实例详解(动力节点Java学院整理)

改成下面这样就行了:

Java类的继承实例详解(动力节点Java学院整理)

  4.super

  super主要有两种用法:

  1)super.成员变量/super.成员方法;

  2)super(parameter1,parameter2….)

  第一种用法主要用来在子中调用父的同名成员变量或者方法;第二种主要用在子的构造器中显示地调用父的构造器,要注意的是,如果是用在子构造器中,则必须是子构造器的第一个语句。

三.常见的面试笔试题

1.下面这段代码的输出结果是什么?

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26
public class test {

public static void main(string[] args) {

new circle();

}

}

class draw {

public draw(string type) {

system.out.println(type+" draw constructor");

}

}

class shape {

private draw draw = new draw("shape");

public shape(){

system.out.println("shape constructor");

}

}

class circle extends shape {

private draw draw = new draw("circle");

public circle() {

system.out.println("circle constructor");

}

}

shape draw constructor

shape constructor

circle draw constructor

circle constructor

  这道题目主要考察的是继承时构造器的调用顺序和初始化顺序。要记住一点:父的构造器调用以及初始化过程一定在子的前面。由于circle的父是shape,所以shape先进行初始化,然后再执行shape的构造器。接着才是对子circle进行初始化,最后执行circle的构造器。

2.下面这段代码的输出结果是什么?

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37
public class test {

public static void main(string[] args) {

shape shape = new circle();

system.out.println(shape.name);

shape.printtype();

shape.printname();

}

}

class shape {

public string name = "shape";

public shape(){

system.out.println("shape constructor");

}

public void printtype() {

system.out.println("this is shape");

}

public static void printname() {

system.out.println("shape");

}

}

class circle extends shape {

public string name = "circle";

public circle() {

system.out.println("circle constructor");

}

public void printtype() {

system.out.println("this is circle");

}

public static void printname() {

system.out.println("circle");

}

}

shape constructor

circle constructor

shape

this is circle

shape

  这道题主要考察了隐藏和覆盖的区别(当然也和多态相关,在后续博文中会继续讲到)。

  覆盖只针对非静态方法(终态方法不能被继承,所以就存在覆盖一说了),而隐藏是针对成员变量和静态方法的。这2者之间的区别是:覆盖受rtti(runtime type identification)约束的,而隐藏却不受该约束。也就是说只有覆盖方法才会进行动态绑定,而隐藏是不会发生动态绑定的。在java中,除了static方法和final方法,其他所有的方法都是动态绑定。因此,就会出现上面的输出结果。

以上所述是小编给大家介绍的java的继承实例详解(动力节点java学院整理),希望对大家有所帮助

收藏 (0) 打赏

感谢您的支持,我会继续努力的!

打开微信/支付宝扫一扫,即可进行扫码打赏哦,分享从这里开始,精彩与您同在
点赞 (0)

声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。

快网idc优惠网 建站教程 Java类的继承实例详解(动力节点Java学院整理) https://www.kuaiidc.com/117667.html

相关文章

发表评论
暂无评论