Java基础深度总结:继承

过去的事,就让它过去吧,我们错过了昨日的日落,再也不能错过今日的日出。

1.继承概述

继承是面向对象思想的三大特性之一,使类与类之间产生特殊 - 一般的关系,即is-a关系。子类继承父类,表名子类是一种特殊的父类,子类拥有父类的属性和方法,并且子类可以拓展具有父类所没有的一些属性和方法,即使子类不拓展父类,也能维持拥有父类的操作。

2.继承的优缺点

优点:

  • 提高代码的复用性、维护性。
  • 让类与类之间产生关系,是多态的前提。

缺点:

  • 继承严重破坏了父类的封装性,子类可以直接访问父类的成员变量(内部信息)和方法, 因此父类封装性被打破。
  • 继承支持扩展,但往往以增强系统结构的复杂度为代价。
  • 类与类之间产生联系,增加了耦合性,子类依赖于父类的实现,子类缺乏独立性。
3.super

super关键字主要存在于子类方法中,用于指向子类对象中父类对象。super的作用主要有两个:

(1)调用父类构造方法

  • 在子类构造函数若没有显示调用父类的构造方法,Java编译器会在子类构造中加上super()语句,默认调用父类的无参构造。
  • 若父类没有无参构造,那么子类构造方法中必须使用super(params)来显示调用父类中的带参构造。
  • super调用父类构造方法的语句必须在子类构造方法中的第一行。
  • super和this关键字不能同时在一个构造函数中调用其他的构造函数,因为这两个语句都要在构造函数的第一句。

(2)调用父类成员(成员变量和方法)

  • super用于限定该对象调用它从父类继承得到的实例变量或方法。若子类含有与父类同名的变量或子类重写了父类中的方法,又想访问父类的,就可以使用super来限定。
class SuperExample {

    protected int x;
    protected int y;

    public SuperExample(int x, int y) {
        this.x = x;
        this.y = y;
    }

    public void func() {
        System.out.println("SuperExample.func()");
    }
}

public class SuperExtendExample extends SuperExample {

    private int z;

    public SuperExtendExample(int x, int y, int z) {
        super(x, y);
        this.z = z;
    }

    @Override
    public void func() {
        super.func();
        System.out.println("SuperExtendExample.func()");
    }
    
    public void func2() {
        System.out.println("x:" + super.x + "--" + "y:" + super.y + "--" + "z:" + this.z);
    }

    public static void main(String[] args) {
        SuperExtendExample s = new SuperExtendExample(1,2,3);
        s.func();
        s.func2();
    }
}

输出:
SuperExample.func()
SuperExtendExample.func()
x:1--y:2--z:3

Tip:

  • super和this不能出现在静态方法中,因为静态方法属于类的,而super和this都与对象相关。
4.Java继承的特点

Java只支持单继承,不支持多继承,但可以多层继承(继承体系)。因为多继承会存在安全隐患,当继承多个存在相同属性或方法的类,子类想调用时就会不知道应该调用哪个类中的属性或方法。

(1)成员变量和方法

  • 子类只能继承父类的所有非私有的成员变量和方法。
  • 子类虽然不能继承父类的私有的成员变量和方法,但是可以通过父类中提供的getter和setter方法,间接的访问和操作父类中的private的属性。
  • 子类中出现了和父类同名的成员变量和成员方法时,父类的成员变量会被隐藏,父类的成员方法会被覆盖。使用super关键字来进行引用父类的成员变量和方法。
  • 当创建一个子类对象时,不仅会为该类的实例变量分配内存,也会为它从父类继承得到的所有实例变量分配内存,即使子类定义了与父类中同名的实例变量。 即依然会为父类中定义的、被隐藏的变量分配内存。

(2)构造器

  • 子类不会继承父类的构造方法。
  • 不要递归调用构造方法,也不要多个构造方法相互循环调用。
5.构造器初始化顺序

父类构造器总是在子类构造器之前执行。存在静态域的初始化顺序参考:Java基础深度总结:关键字-static

6.使用继承的注意事项

为了保证父类有良好的封装性,不会被子类随意改变,设计父类通常应该遵循如下规则:

(1)尽量隐藏父类的内部数据。

      尽量把父类的所有成员变量都设置成private访问类型,不要让子类直接访问父类的成员变量。

(2)不要让子类随意访问、修改父类的方法。

  • 父类中那些仅为辅助其他的工具方法,应该使用private修饰,让子类无法访问方法;

  • 如果父类中的方法需要被外部类调用,则必须以public修饰,但又不想让子类重写,就可以使用final修饰符。

  • 如果希望父类的某个方法被子类重写,但不希望被其他类自由访问,则可以使用protected来修饰方法。.

(3)不要在父类构造器中调用将要被子类重写的方法。

class Base{

    public Base(){
        test();
    }

    public void test(){
        System.out.println("将被子类重写的方法");
    }
}

public class Sub extends Base {

    private String name="aa";

    public Sub(){
    }

    public void test(){
        System.out.println("子类重写父类的方法,"
                + "其name字符串长度" + name.length());
    }

    public static void main(String[] args){
        // 空指针异常
        Sub s = new Sub();
    }
}

解析:
当创建Sub对象时,先执行其父类构造器,子类重写了test方法,就会执行子类中的test()方法,子类的test方法调用了子类的实例变量name,而此时子类对象还未初始化,所以name为null,name.length()引发空指针异常。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值