零基础Java第十三期:继承与多态(一)
目录
一、继承
1.1. 继承的目的
1.2. 继承的概念
1.3. 继承的语法
1.4. 父类的访问
1.5. 继承中的重载与重写
1.6. 子类的构造方法
1.7. 再谈初始化
一、继承
1.1. 继承的目的
我们来定义一个Dog和Cat的类:
public class Dog {
public int age;
public String breed;
public String name;
public void eat(){
System.out.println("在吃饭");
}
}
public class Cat {
public int age;
public String breed;
public String name;
public void eat(){
System.out.println("在吃饭");
}
}
我们可以看到这两个类当中的成员变量是相同的,那么我们能不能把相同的成员变量抽取出来,放到同一个类中呢?面向对象思想中提出了继承的概念,专门用来进行共性抽取,实现代码复用。
public class Animal {
public int age;
public String breed;
public String name;
public void eat(){
System.out.println("在吃饭");
}
}
1.2. 继承的概念
继承机制:允许程序员在保持原有类特 性 的基础上进行扩展,增加新功能,这样产生新的类,称派生类。通俗点说,可以看成是…… is …… a 的关系。
继承呈现了面向对象程序设计的层次结构, 体现了 由简单到复杂的认知过程。继承主要解决的问题是:共性的抽取,实现代码复用。
1.3. 继承的语法
语法规则:
修饰符 static 子类(派生类) extends 父类(基类、超类){
public static Cat extends Animal{
public static Dog extends Animal{
1.4. 父类的访问
(1)父类与子类的成员变量不同名时
public class Base {
public int a = 1;
public int b = 2;
}
public class Derived extends Base{
public int c = 3;
public void test(){
System.out.println(a);
System.out.println(b);
System.out.println(c);
}
public static void main(String[] args){
Derived num = new Derived();
num.test();
}
}
(2)父类与子类的成员变量同名时
public class Base {
public int a = 1;
public int b = 2;
}
public class Derived extends Base{
public int a = 10;
public int c = 3;
public void test(){
System.out.println(a);
System.out.println(b);
System.out.println(c);
}
public static void main(String[] args){
Derived num = new Derived();
num.test();
}
}
此时就会优先访问子类成员变量。那如果我就想访问父类的成员变量呢?我们就可以引入一个super关键字。
public class Base {
public int a = 1;
public int b = 2;
}
public class Derived extends Base{
public int a = 10;
public int c = 3;
public void test(){
System.out.println(super.a);
System.out.println(b);
System.out.println(c);
}
public static void main(String[] args){
Derived num = new Derived();
num.test();
}
}
下面博主将会画图来给大家讲解一下原理:
父类与子类的成员变量都可以通过this来进行访问,而父类中的成员变量也可以通过super来访问。所以说,我们在访问成员变量b时,也可以在前面加上super关键字,这样就能提高代码的可读性,以至于不会搞混成员变量到底是父类的还是子类的。
我们可以总结下来就是:1、当访问不同的成员时,优先看父类有没有;2、父类有访问父类的,父类没有,看子类有没有;3、子类有,访问子类的,如果没有,就会报错。
1.5. 继承中的重载与重写
public class Base {
public int a = 1;
public int b = 2;
public void Basefunc(){
System.out.println("base::func");
}
}
public class Derived extends Base{
public void Derivedfunc(){
System.out.println("derived::func");
}
public void Basefunc(){
System.out.println("base::func");
}
public void Basefunc(int a){
System.out.println(a);
}
public void Method(){
Derivedfunc();
Basefunc();
Basefunc(20);
}
public static void main(String[] args) {
Derived num = new Derived();
num.Method();
}
}
在上面的代码中我们可以看到父类与子类中,存在名称相同的方法。父类中的Basefunc方法与子类中的Derivedfunc方法构成了方法重写;而与子类中的Derivedfunc(a)构成了方法重载。
1.6. 子类的构造方法
父子父子,先有父再有子,即:子类对象构造时,需要先调用父类构造方法,然后执行子类的构造方法。 在构造子类对象时候 ,先要调用父类的构造方法,将从父类继承下来的成员构造完整 ,然后再调用子类自己的构造方法,将子类自己新增加的成员初始化完整 。
public class Animal {
public int age;
public String name;
public Animal(int age, String name) {
this.age = age;
this.name = name;
}
public void eat(){
System.out.println("在吃饭");
}
}
public class Cat extends Animal{
public void sleep(){
System.out.println("睡觉");
}
public Cat(String name,int age){
super(age, name);
}
}
public class Dog extends Animal{
public void bark(){
System.out.println("汪汪叫");
}
public void wag(){
System.out.println("摇尾巴");
}
public Dog(String name,int age){
super(age, name);
}
}
public class TestDemo {
public static void main(String[] args) {
Cat cat = new Cat("小黑",5);
cat.name = "大黄";
cat.eat();
System.out.println("======");
Dog dog = new Dog("大黄",10);
dog.name = "小黑";
dog.bark();
}
}
如果我们对TestDemo里的类进行调试,可以看到在父类里面进行传参和赋值。
注意:(1)若父类显式定义无参或者默认的构造方法,在子类构造方法第一行默认有隐含的super()调用,即调用父类构造方法。 如果我们在父类里面再写一个构造方法,则子类里面就会报错。
//父类里面默认的一个构造方法
public Animal(int age, String name) {
this.age = age;
this.name = name;
}
(2)如果父类构造方法是带有参数的,此时需要用户为子类显式定义构造方法,并在子类构造方法中选择合适的 父类构造方法调用,否则编译失败。
(3) 在子类构造方法中,super(...)调用父类构造时,必须是子类构造函数中第一条语句。
1.7. 再谈初始化
继承关系上的执行顺序
public class Person {
public String name;
public int age;
public Person(String name, int age) {
this.name = name;
this.age = age;
}
{
System.out.println("Person:实例代码块执行");
}
static {
System.out.println("Person:静态代码块执行");
}
}
public class Student extends Person{
public Student(String name,int age) {
super(name,age);
System.out.println("Student:构造方法执行");
}
{
System.out.println("Student:实例代码块执行");
}
static {
System.out.println("Student:静态代码块执行");
}
}
所以,继承的顺序为:1、父类静态代码块优先于子类静态代码块执行,且是最早执行;2、父类实例代码块和父类构造方法紧接着执行;3、子类的实例代码块和子类构造方法紧接着再执行;4、第二次实例化子类对象时,父类和子类的静态代码块都将不会再执行。