面对对象---多态性、方法的重写

一、重点知识

记得有public class时候 main函数在public class内执行,不然你run as时候找不到

一个类中可以没有public class

接口也可以继承,也用extends

每个对象的属性是独立的,存储在堆空间中,就会自动给默认值,类比数组,而局部变量存放在栈内存中,无默认值

接口中是不能调用代码块的

接口也算引用类型,也可以定义变量

实例化子类对象的过程
调用父类的构造器(默认调用无参构造)后再调用子类的构造
走子类得构造一定会先走父类的构造,创建空间存储父类的属性。
子类构造器如果形参列表多于构造器中赋值的属性,不会报错,但是也赋不进去值,打印显示会是默认初始化的值。少于构造器中中的赋值,则会报错

一般子类无参构造器,不会调用父类的有参,因为没有意义

开发中子类与父类中不建议定义同名属性,可以但没必要

this和super对重写的方法同样适用,这里可以看出,父类的被重写的方法并不是被覆盖了,他仍然存在,只是默认不调用它,创建的谁的对象就调用谁的,注意多态中的规则

静态方法中不能出现this 和super关键字

导入包时候.* 和.具体的包效率上是没有区别的

二、重点问题

1、说明抽象类也可以实现接口问题

Java开发中,接口通过让类去实现(implements)的方式来使用.
如果实现类覆盖了接口中的所有抽象方法,则此实现类就可以实例化
如果实现类没有覆盖接口中所有的抽象方法,则此实现类仍为一个抽象类

2、super的梳理与使用

public class HomeWork{

public static void main(String[] args) {

B test = new B(19,"name",18);

System.out.println(test.age);

System.out.println(test.name);

System.out.println(test.sum);

}

}

class A{

int age;

String name;

public A(){

}

public A(int age,String name){

this.age = age;

this.name = name;

}

}

class B extends A{

int sum;

public B(int age, String name,int sum) {

super(18, "张山");

// this.sum = sum;

}

}

总结:如果你写了参数,里面没有定义的赋值的话,赋不上值,打印输出的话仍为默认值,如果你用super(参数)调用父类构造器时,参数的值被你写过了,那么子类创建对象的属性就是你调用的supers时固定的属性,不会再变化,即使你后来创造对象时候再赋值,也不会再改变
注意super可以指代对象,不然他怎么能通过.对象 .方法的格式来调用呢?super用于参数会报错是因为,虚拟机觉得super作为参数没有意义,所以会报错 ,在子类中直接打印输出this,显示的为此类的地址


三、课堂知识

3.1、子类对象的创建过程

实例化子类对象的过程:

1、先调用父类的构造方法

默认调用父类的无参构造方法。

2、再调用子类的构造方法

我们通过代码演示,发现当创建子类对象的时候,默认会先调用父类的构造方法。

调用子类的构造方法中,先调用了父类的构造方法,默认调用的父类的无参构造方法。

1、子类的构造方法中,第一行默认是调用了父类的构造方法。

2、如果没有显示的调用父类的构造方法,那么编译器会自动添加一个:super(),父类的无参构造方法。

3、子类的构造方法中,既可以调用父类的无参构造方法,也可以调用父类的有参构造方法。

4、父类的构造方法:

    无参构造:super();

有参构造:super(参数);

3.2、super关键字

用法一:super(),指代父类的构造方法。

如果有多个父类的构造方法,那么要靠参数区分。

super(),无参构造

super(参数),父类的有参构造方法。。

注意点:super(),指代父类的构造方法,一定是位于第一行。同this()指代本类构造方法。

方法二:直接访问父类的属性和方法

super.属性--->父类的属性。

super.方法--->父类的方法

3.3、方法的重写

重载(overload):一个功能方法的多种写法。

同一个类中,方法名必须一致,参数列表必须不同。

重写(override):继承结构。

概念:在继承结构中,子类将父类已有的方法,重新实现,这叫方法的重写。

如果在子类中,存在方法的重写,那么默认调用的是重写之后的方法。

方法重写的规则:

1、继承结构中。

2、方法的声明,必须和父类的一致。

方法名必须和父类一致

参数列表必须和父类一致

基本数据类型的返回值必须和父类一致,应用类型的返回值可以是父类引用类型返回值的子类

3、访问权限,不能比父类更加严格。

4、子类重写的方法,不能抛出比父类更大更多的异常。

3.4、访问权限修饰符

访问权限:就是在哪能访问。通过访问权限修饰符来实现。

private:私有的。本类中。

默认的:(default):注意,什么修饰符都不写,就是默认的。本类。同包下可以访问。

protected:受保护的。本类,同包下,不同包(子类使用)

public,没有限制的。

private---> default--->protected--->public

使用原则:

修饰类:只能是public和default

修饰成员:变量和方法和构造器和内部类:以上4个修饰符都可以。

局部变量,不能使用这些修饰符。

3.5、对象的多态性

1 什么是多态

其实就是事物的多种形态。

2.何为多态性:

对象的多态性:父类的引用指向子类的对象(或子类的对象赋给父类的引用)

3.多态性的使用:虚拟方法调用

> 有了对象的多态性以后,我们在编译期,只能调用父类中声明的方法,但在运行期,我们实际执行的是子类重写父类的方法。

> 总结:编译,看左边;运行,看右边。

4.多态性的使用前提:

① 类的继承关系  ② 方法的重写

6.多态性使用的注意点:

对象的多态性,只适用于方法,不适用于属性(编译和运行都看左边)

3.6、向上转型和向下转型

向上转型:一个子类对象看成了父类类型。那么就失去了子类新增。

向上转型:多态

 向下转型:

1 为什么使用向下转型:

有了对象的多态性以后,内存中实际上是加载了子类特有的属性和方法的,但是由于变量声明为父类类型,导致编译时,只能调用父类中声明的属性和方法。子类特有的属性和方法不能调用。如何才能调用子类特的属性和方法?使用向下转型。

2 如何实现向下转型:

使用强制类型转换符:()

3 使用时的注意点:

① 使用强转时,可能出现ClassCastException的异常。

② 为了避免在向下转型时出现ClassCastException的异常,我们在向下转型之前,先进行instanceof的判断,一旦返回true,就进行向下转型。如果返回false,不进行向下转型。

4 instanceof的使用:

① a instanceof A:判断对象a是否是类A的实例。如果是,返回true;如果不是,返回false。

② 如果 a instanceof A返回true,则 a instanceof B也返回true.其中,类B是类A的父类。

你可能感兴趣的:(面对对象---多态性、方法的重写)