夯实Java基础(三)----面向对象之继承

JAVA面向对象三大特性,封装继承多态

一、继承的定义

1、JAVA继承定义:

子类继承父类的特征和行为,使得子类对象具有父类相同的特征和行为,继承描述的是事物之间的所属关系,这种关系是:is-a 的关系。

子类,也叫派生类;被继承的称为父类、超类(superclass)或者基类。

2、JAVA继承目的:

如果多个子类中存在相同的特征与行为,那么就可以继承父类,父类中的特征和行为可以被子类继承下来使用,则不需要在子类中重复定义。

3、继承的特点:

①、Java是单继承,不可以多继承,但可以多层继承。

②、子类只能继承父类非private的属性,方法。(其实子类继承父类后,仍然认为获取了父类的private结构,因为封装的影响,使得子类不能直接调用父类的private结构而已,setter或getter)。

③、子类可以对父类进行扩展,也可以用自己的方式继承父类的方法(重写)。

④、提高了类之间的耦合性(继承的缺点,耦合度高就会造成代码之间的联系)。

4、继承的优点

减少代码的冗余,提高代码的复用性。

便于功能的扩展。

为后面多态提供前提。

5、继承的缺点

子类与父类存在严重的耦合关系

继承破坏了父类的封装性

子类继承父类的属性和方法,也就说明可以从子类中恶意修改父类的属性和方法。

6、类的继承格式

通过extends关键字实现继承关系

[修饰符] class 父类 {
    属性、方法
}
[修饰符] class 子类 extends 父类 {
    属性、方法
}

7、何时使用继承

①、子类需要额外增加属性,而不仅仅是属性值的改变。

②、子类需要增加自己独有的行为方式。(包括增加新的方法或重写父类的方法)

二、构造器

1、构造器定义:

构造器也叫构造方法或构造函数,作用是用于创建并初始化对象。

注意:构造器只为实例变量初始化,不为静态类变量初始化

2、无参构造与有参构造

当我们在使用new关键字创建对象时,A a = new A(); 此时使用了无参构造器创建了一个对象,创建完成后它的成员变量也会被初始化,默认是相对数据类型的默认值。如果我们需要为成员变量赋值,则再挨个为它们再赋值,太麻烦了。使用有参构造就可以在new对象时,直接为当前对象的某个或所有成员变量直接赋值。

3、构造器语法格式

[修饰符] 构造器名(){
    // 实例初始化代码
}
[修饰符] 构造器名(参数列表){
    // 实例初始化代码
}

构造器创建规则:

①、构造器名必须与类名一致。

②、构造器无返回值,不需要返回值类型,不需要void。

③、系统给出默认无参构造器,并该构造器的修饰符默认与类修饰符相同。

④、如果写了构造器,无论是有参无参,系统都不再提供无参构造。

⑤、构造器可以重载,既可以定义参数,也可以不定义参数。

public class Father {
    private String name;
    private int age;
    // 无参构造
    public Father() {}
    // 有参构造
    public Father(String name,int age) {
        this.name = name;
        this.age = age;
    }
    //getter,setter方法省略...
}

三、方法重载与重写

1、定义:

方法重载(overload):指在同一个类中,允许存在一个以上的同名方法,只要参数列表不同即可。

注意:参数列表不同,指的是参数个数不同,数据类型不同,数据类型顺序不同。

方法重写(overrid):子类中继承父类的某个方法方法,但并不能满足实际需求,那么则需要方法重写,覆盖父类的方法。方法体不同,用于改造并覆盖。

2、重写规则:

①、保证父子类间方法的名称相同,参数列表也相同

②、父子类间的返回值类型相同,或者为父类返回值类型的子类类型。

③、子类方法的访问修饰符不能小于父类。(public>protected>default>private)

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

静态方法不能被重写,方法重写指的是实例方法重写,静态方法属于类方法不能被重写,而是隐藏

在子类中不可见的方法(私有方法等)不能被重写。

final方法不能被重写。

四、this和super关键字

1、this关键字

①、含义:

表示当前对象的引用。可用于调用本类的属性、方法、构造器。

②、使用位置:

this在实例初始化相关代码块和构造器中,表示正在创建的实例对象,即正在new谁this就代表谁。

tnis在非静态实例方法中,表示调用该方法的对象,即谁在调用this就代表谁。

this不能出现在静态代码块和静态方法中。

③、this使用格式:

this.成员变量名:当方法的局部变量与当前对象重名时,就可以在成员变量前加this.。

this.成员方法:调用当前对象的成员方法时,都可以加this,也可以省略,实际开发都省略。

this.调用都是先从本类声的成员变量或方法中查找,如果未找到,会去从父类继承的子类中查找。

③、this()或this(实参列表)

只能调用本类的其他构造器

必须在构造器的首行

如果一个类中声明了n个构造器,则最多有n-1个构造器使用了this(实参列表),否则会发生递归调用死循环。

/**
 * 通过this调用构造器测试
 */
public class Demo {
    public static void main(String[] args) {
        Father father = new Father("张三", 30);
        father.show();
    }
}

class Father {
    public String name;
    public int age;

    //1、无参构造器
    public Father() {
        System.out.println("调用了Father无参构造器");
    }

    //2、name属性的有参构造器
    public Father(String name) {
        this();
        this.name = name;
        System.out.println("调用了Father的name属性有参构造器");
    }

    //3、全参构造器
    public Father(String name, int age) {
        this(name); //调用2
        this.name = name; 
        this.age = age;
        System.out.println("调用了Father的全参构造器");
    }

    // 定义show方法
    public void show() {
        System.out.println("name:" + this.name + ",age:" + this.age);
    }
}

2、super关键字

①、含义:

super是用于当前类中访问父类的一个特殊关键字,可用于调用父类的属性、方法、构造器,它不是对象的引用。

②、使用格式:

(1)super.成员变量:在子类中访问父类的成员变量,如果子类的成员变量与父类的成员变量重名时,必定是调用父类的成员变量。

(2)super.成员方法:在子类中调用父类的成员方法,如果子类重写了父类的成员方法,必定是调用父类的成员方法。

(3)super()或super(实参列表):在子类的构造器首行,用于表示调用父类的哪个实例初始化方法super()默认调用父类构造器。

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

public class Demo {
    public static void main(String[] args) {
        Son son = new Son();
        son.show();
    }
}
class Father {
    String name = "Father";
    int age = 40;

    public Father() {
        System.out.println("调用了父类的无参构造器");
    }
    public Father(String name, int age) {
        this.name = name;
        this.age = age;
    }
    public void show() {
        System.out.println("父类的Show方法");
    }
}
class Son extends Father {
    String name = "Son";
    int age = 20;

    public Son() {
        //调用父类构造器,可以省略,系统会默认添加
        super();
    }
    public void show() {
        System.out.println("子类的Show方法");
        System.out.println("本类属性:" + this.name + "," + this.age);
        //调用父类方法
        super.show();
        System.out.println("父类属性:" + super.name + "," + super.age);
    }
}

3、this、super小结

使用this、super调用构造器时,必须放在构造方法的第一行,所以不能同时出现,否则编译报错。

子类会默认使用super调用父类默认构造器,如果父类没有默认构造器,则子类必须显式的使用super()来调用父类构造器。

只能使用super关键字来调用父类的构造方法,不能使用父类构造方法名来调用,因为父类的构造方法不能被子类继承。

如果子类中没显式调用,则编译器自动调用super(),也就是说一直调到Object类,因为Object是所有类的父类。

super和this都不能出现在静态方法和静态代码块中,因为super和this都是存在于对象中的。

4、this、super联系

①、父类,子类及子类方法中存在同名变量时

public class Test {
    public static void main(String[] args) {
        Son son = new Son();
        System.out.println(son.a);//20
        System.out.println(son.b);//11
        son.test();
        son.method(30);
        son.fun(13);
    }
}

class Father {
    int a = 10;
    int b = 11;
}
class Son extends Father {
    int a = 20;

    public void test() {
        //子类与父类的属性同名,子类对象中就有两个a
        System.out.println("父类的a:" + super.a);//10    直接从父类局部变量找
        System.out.println("子类的a:" + this.a);//20   先从本类成员变量找
        System.out.println("子类的a:" + a);//20  先找局部变量找,没有再从本类成员变量找

        //子类与父类的属性不同名,是同一个b
        System.out.println("b = " + b);//11  先找局部变量找,没有再从本类成员变量找,没有再从父类找
        System.out.println("b = " + this.b);//11   先从本类成员变量找,没有再从父类找
        System.out.println("b = " + super.b);//11  直接从父类局部变量找
    }
    public void method(int a) {
        //子类与父类的属性同名,子类对象中就有两个成员变量a,此时方法中还有一个局部变量a
        System.out.println("父类的a:" + super.a);//10  直接从父类局部变量找
        System.out.println("子类的a:" + this.a);//20  先从本类成员变量找
        System.out.println("局部变量的a:" + a);//30  先找局部变量
    }
    public void fun(int b) {
        System.out.println("b = " + b);//13  先找局部变量
        System.out.println("b = " + this.b);//11  先从本类成员变量找
        System.out.println("b = " + super.b);//11  直接从父类局部变量找
    }
}

②、父子类中找方法

public class Test {
    public static void main(String[] args) {
        Son s = new Son();
        System.out.println(s.getNum());//10   没重写,先找本类,没有,找父类

        Daughter d = new Daughter();
        System.out.println(d.getNum());//20  重写了,先找本类
    }
}
class Father {
    protected int num = 10;

    public int getNum() {
        return num;
    }
}
class Son extends Father {
    private int num = 20;
}

class Daughter extends Father {
    private int num = 20;

    public int getNum() {
        return num;
    }
}

③、父子类中找方法

public class Test {
    public static void main(String[] args) {
        Son s = new Son();
        s.test();

        Daughter d = new Daughter();
        d.test();
    }
}
class Father {
    protected int num = 10;

    public int getNum() {
        return num;
    }
}
class Son extends Father {
    private int num = 20;

    public void test() {
        System.out.println(getNum());//10  本类没有找父类
        System.out.println(this.getNum());//10  本类没有找父类
        System.out.println(super.getNum());//10  本类没有找父类
    }
}
class Daughter extends Father {
    private int num = 20;

    public int getNum() {
        return num;
    }
    public void test() {
        System.out.println(getNum());//20  本类有,先找本类
        System.out.println(this.getNum());//20  本类有,先找本类
        System.out.println(super.getNum());//10  重写了,直接找父类
    }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值