Java学习笔记之继承与修饰符

0x00 概述

本文涉及Java知识点为继承和修饰符

 

0x01 继承

1.1 继承的实现

  • 继承的概念

         继承是面向对象三大特征之一,可以使得子类具有父类的属性和方法,还可以在子类中重新定义,以及追加属性和方法。

  • 实现继承的格式

         继承通过extends实现

         格式:class 子类 extends 父类 {}

                 举例:class dog extends Animal {}

  • 继承带来的好处:

         继承可以让类与类之间产生关系,子父关系,产生子父类后,子类则可以使用父类中非私有的成员。

示例

package com.superTest1;

public class Fu {
    public void show() {
        System.out.println("Fu show()方法被调用");
    }
}
package com.superTest1;

public class Zi extends Fu{
    public void method() {
        System.out.println("Zi method方法被调用");
    }
}
package com.superTest1;

public class Demo {
    public static void main(String[] args) {
        // 创建对象,调用方法
        Fu f = new Fu();
        f.show();

        Zi z = new Zi();
        z.method();
        z.show();
    }
}

输出结果

/*

Fu show()方法被调用
Zi method方法被调用
Fu show()方法被调用

*/

 

1.2 继承的好处和弊端

  • 继承的好处

        提供了代码的复用性(多个类相同的成员可以放到同一个类中)

         提高了代码的维护性(如果方法的代码需要修改,修改一处即可)

  • 继承的弊端

         继承让类与类之间产生了关系,类的耦合性增强了,当父类发生变化时子类实现也不得不跟着变化,削弱了子类的独立性

  • 继承的应用场景

         使用继承,需要考虑类与类之间是否存在is a的关系,不能盲目使用继承

         is a的关系,谁是谁的一种,例如:老师和学生是人的一种,那人就是父类,学生和老师都是子类

 

0x02 继承中成员访问特点

2.1 继承中变量的访问特点

在子类方法中访问一个变量,采用的就是就近原则

1. 子类局部范围找

2. 子类成员范围找

3. 父类成员范围找

4. 如果都没有就报错(不考虑父类的父类)

示例

package com.superTest2;

public class Fu {
    int num = 10;
}
package com.superTest2;

public class Zi extends Fu {
    int num = 20;

    public void show() {
        int num = 30;
        System.out.println(num);
    }
}
package com.superTest2;

public class Demo {
    public static void main(String[] args) {
        Zi z = new Zi();
        z.show();
    }
}

输出

30

 

2.2 super

  • this & super关键字:

         this:代表本类对象的引用

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

  • this & super的使用分别:

         成员变量:

               this.成员变量 - 访问本类成员变量

               super.成员变量 - 访问父类成员变量

          成员方法:

               this.成员方法 - 访问本类成员方法

       super.成员方法 - 访问父类成员方法

  • 构造方法:

       this(...) - 访问本类构造方法

       super(...) - 访问父类构造方法

 

2.3 继承中构造方法的访问特点

注意:子类中所有的构造方法默认都会访问父类中无参的构造方法,

子类会继承父类中的数据,可能还会使用父类的数据,所以,子类初始化之前,一定要先完成父类数据的初始化,

原因在于,每一个子类构造方法的第一条默认都是super(), 继承超类Object(Object类可自行百度)

问题:

如果父类中没有无参构造方法,只要带参构造方法,该怎么办?

/*

1 通过使用super关键字去显式的调用父类的带参构造方法
2 在父类中自己提供一个无参构造方法

*/

推荐方案:

自己给出无参构造方法

 

2.4 继承中成员方法的访问特点

通过子类对象访问一个方法

1. 子类成员范围找

2. 父类成员范围找

3. 如果都没有就报错(不考虑父类的父类)

 

2.5 super内存图

  • 对象在堆内存内,会单独存在一块super区域,用来存放父类的数据

 

2.6 方法重写

1. 方法重写概念

  子类出现了和父类中一模样的方法声明(方法名一样,参数列表也必须一样)

2. 方法重写的应用场景

  当子类需要父类的功能,而功能主体类有自己特有的内容时,可以重写父类中的方法,这样,既沿袭了

  父类的功能,有定义了子类特有的内容

3. Override注解

  用来检测当前的方法,是否为重写的方法,起到了校验的作用

 

2.7 方法重写的注意事项

方法重写注意事项:

1. 私有方法不能被重写(父类私有成员子类时不能继承的)

2. 子类方法访问权限不能更低(public>默认>私有)

示例

package com.superTest3;

public class Fu {
    private void show() {
        System.out.println("Fu中show()方法被调用");
    }

    void method() {
        System.out.println("Fu中method()方法被调用");
    }
}
package com.superTest3;

public class Zi extends Fu {
    /*
    // 编译失败,子类重写父类方法的时候,不能重写父类私有的方法
    @Override
    public void show() {
        System.err.println("Zi中show()方法被调用");
    }


    // 编译失败,子类重写父类方法的时候,访问权限需要大于等于父类
    @Override
    private void method() {
        System.out.println("Zi中method()方法被调用");
    }

     */

    @Override
    public void method() {
        System.out.println("Zi中method()方法被调用");
    }
}

 

2.8 Java中继承的注意事项

Java中集成的注意事项

1. Java中类只支持单继承,不支持多继承

  错误范例:class A extends B,C {}

2. Java中类支持多层继承

示例

package com.superTest4;

public class Granddad {
    public void drink() {
        System.out.println("爷爷爱喝酒");
    }
}
package com.superTest4;

public class Father extends Granddad{
    public void smoke() {
        System.out.println("爸爸爱抽烟");
    }
}
package com.superTest4;

public class mother {
    public void dance() {
        System.out.println("妈妈爱跳舞");
    }
}
package com.superTest4;

public class Son extends Father {
    // 此时son类中同时拥有drink和smoke方法
}

 

0x03 继承练习

3.1 老师和学生

需求:定义老师类和学生类,然后写代码进行测试;最后找到老师类和学生类当中的共性内容,抽取一个父类,用继承的方式改写代码,并进行测试

步骤:

1. 定义老师类(姓名,年龄,教书())

2. 定义学生类(姓名,年龄,学习())

3. 定义测试类,写代码进行测试

4. 共性抽取父类,定义人类(姓名,年龄)

5. 定义老师类,继承人类,并给出自己特有的方法,教书()

6. 定义学生类,继承人类,并给出自己特有的方法,学习()

7. 定义测试类,写代码测试

示例:

package com.extendTest1;

public class Person {
    private String name;
    private int age;

    public Person() {
    }

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }
}
package com.extendTest1;

class Teacher extends Person {
    public Teacher() {
    }

    public Teacher(String name, int age) {
        super(name, age);
    }

    public void teach() {
        System.out.println("老师会教书");
    }
}
package com.extendTest1;

public class Student extends Person {
    public Student() {
    }

    public Student(String name, int age) {
        super(name, age);
    }

    public void study() {
        System.out.println("学习要学习");
    }

}
package com.extendTest1;

public class PersonDemo {
    public static void main(String[] args) {
        // 创建老师类对象并进行测试
        Teacher t1 = new Teacher();
        t1.setName("Alice");
        t1.setAge(33);
        System.out.println(t1.getName() + ", " + t1.getAge());
        t1.teach();

        Teacher t2 = new Teacher();
        t2.setName("Bob");
        t2.setAge(25);
        System.out.println(t2.getName() + ", " + t2.getAge());
        t1.teach();

        // 创建学生类对象进行测试
        Student s1 = new Student("Charlie", 12);
        System.out.println(s1.getName() + ", " + s1.getAge());
        s1.study();
    }
}

 

3.2 猫和狗

需求:请采用继承的思想实现猫和狗的案例,并在测试类中进行测试

分析:

1. 猫:

  成员变量:姓名,年龄

  构造方法:无参,带参

  成员方法:get/set方法,抓老鼠()

2. 狗

  成员变量:姓名,年龄

  构造方法:无参,带参

  成员方法:get/set方法,看门()

步骤:

1. 定义动物类(Animal)

  成员变量:姓名,年龄

  构造方法:无参,带参

  成员方法:get/set方法

2. 定义猫类(Cat),继承动物类

  构造方法:无参,带参

  成员方法:get/set方法,抓老鼠()

3. 定义狗类(Dog),继承动物类

  构造方法:无参,带参

  成员方法:get/set方法,看门()

4. 定义测试类(AnimalDemo),写代码测试

示例

package com.extendTest2;

public class Animal {
    private String name;
    private int age;

    public Animal() {
    }

    public Animal(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }
}
package com.extendTest2;

public class Cat extends Animal {
    public Cat() {
    }

    public Cat(String name, int age) {
        super(name, age);
    }

    public void catchMouse() {
        System.out.println("猫捉老鼠");
    }
}
package com.extendTest2;

public class Dog extends Animal {
    public Dog() {
    }

    public Dog(String name, int age) {
        super(name, age);
    }

    public void lookDoor() {
        System.out.println("狗看门");
    }
}
package com.extendTest2;

public class AnimalDemo {
    public static void main(String[] args) {
        // 创建猫类对象进行测试
        Cat c1 = new Cat();
        c1.setName("Garfileds");
        c1.setAge(2);
        System.out.println(c1.getName()+", "+c1.getAge());
        c1.catchMouse();

        // 创建狗类对象进行测试
        Dog d1 = new Dog();
        d1.setName("wangwang");
        d1.setAge(4);
        System.out.println(d1.getName()+", "+d1.getAge());
        d1.lookDoor();
    }
}

 

0x04 修饰符

4.1 package

1. 包的概念

  包就是文件夹,用来管理类文件的

2. 包的定义格式

  package包名;(多级包用 . 分开)

  例如:package.com.heima.demo;

3. 带包编译&带包运行

  带包编译: java -d .类名.java

    例如: java -d  .com.heima.demo.HelloWorld.java

  带包运行: java 包名+类名

    例如:java com.heimademo.HelloWorld

 

4.2 import

  • 导包的意义

    使用不同包下的类时,使用的时候要写类的全路径,写起来太麻烦了

    为了简化带包操作,java就提供了导包的功能

  • 导包的格式

    格式:import 包名;

    范例: import java.util.Scanner;

示例(没有使用导包,创建的Scanner对象)

package com.heima;

public class Demo {
    public static void main(String[] args) {
        // 1 没有导包,创建Scanner对象
        java.util.Scanner sc = new java.util.Scanner(System.in);
    }
}    

示例(使用导包,创建Scanner对象)

package com.heima;

import java.util.Scanner;

public class Demo {
    public static void main(String[] args) {
        // 1 使用导包,创建Scanner对象
        Scanner sc = new Scanner(System.in);
    }
} 

 

4.3 权限修饰符

 

 

4.4 final

  • final关键字的作用

    final代表最终的意思,可以修饰成员方法,成员变量,类

  • final修饰类,方法,变量的效果

    final修饰类,该类不能被继承,不能有子类,但是可以有父类

    final修饰方法,该方法不能被重写

    final修饰变量,表明该变量是个常量,不能再次赋值

 

4.5 final修饰局部变量

  • final修饰基本数据类型变量

    final修饰指的是节本类型的数据值不能发生改变

  • final修饰引用数据类型变量

    final修饰指的是引用类型地址值不能发生改变,但是地址里面的内容是可以发生改变的

示例

public static void main(Sring[] args) {
    final Student s = new Student(23);
    s = new Student(25);
    s.setAge(26) ;   
}

 

4.6 static

  • static的概念

    static关键字是静态的意思,可以修饰成员方法,成员变量

  • static修饰的特点

    1. 被类的所有对象共享,这也是我们判断是否使用静态关键字的条件

    2. 可以通过类名调用,当然也可以通过对象名调用(推荐使用类名调用)

示例

package com.staticDemo;

public class Student {
    public String name;
    public int age;

    // 学校为共享数据,所以设计为静态
    public static String university;

    public void show() {
        System.out.println(name + ", "+ age + "," + university );
    }
}
package com.staticDemo;

public class staticDemo {
    public static void main(String[] args) {
        // 为对象的共享值
        Student.university = "XX大学";

        Student s1 = new Student();
        s1.name = "Alice";
        s1.age = 30;
        s1.show();

        Student s2 = new Student();
        s2.name = "Bob";
        s2.age = 32;
        s2.show();

    }

}

 

4.7 static访问特点

 static的访问特点

  • 非静态的成员方法

    能访问静态的成员变量

    能访问非静态的成员变量

    能访问静态的成员方法

    能访问非静态的成员方法

  • 静态的成员方法

    能访问静态的成员变量

    能访问静态的成员方法

  • 总结:

    静态成员方法智能访问静态成员

 

posted @ 2021-11-29 09:03  时光飞逝,逝者如斯  阅读(96)  评论(0编辑  收藏  举报