java基础------------------继承

1、子类拥有父类非private的属性和方法。

          2、子类可以拥有自己属性和方法,即子类可以对父类进行扩展。

         3、子类可以用自己的方式实现父类的方法。以后介绍)。

       综上所述,使用继承确实有许多的优点,除了将所有子类的共同属性放入父类,实现代码共享,避免重复外,还可以使得修改扩展继承而来的实现比较简单。

       诚然,讲到继承一定少不了这三个东西:构造器、protected关键字、向上转型。

一。构造器

  子类是不能够继承父类的构造器,但是要注意的是,如果父类的构造器都是带有参数的,则必须在子类的构造器中显示地通过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
23
24
25
26
27
28
29
30
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;
     }
}

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

  改成下面这样就行了:

  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
27
28
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
38
39
40
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方法,其他所有的方法都是动态绑定。因此,就会出现上面的输出结果。

注:static方法,具体的原理我也说不太清。不过根据网上的资料和我自己做的实验可以得出结论:static方法可以被子类继承,但是不能被子类重写(覆盖),但是可以被子类隐藏。(这里意思是说如果父类里有一个static方法,它的子类里如果没有对应的方法,那么当子类对象调用这个方法时就会使用父类中的方法。而如果子类中定义了相同的方法,则会调用子类的中定义的方法。唯一的不同就是,当子类对象上转型为父类对象时,不论子类中有没有定义这个静态方法,该对象都会使用父类中的静态方法。因此这里说静态方法可以被隐藏而不能被覆盖。这与子类隐藏父类中的成员变量是一样的。隐藏和覆盖的区别在于,子类对象转换成父类对象后,能够访问父类被隐藏的变量和方法,而不能访问父类被覆盖的方法



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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值