多态:将一个子类对象赋给父类的引用变量;
最终多态体现为父类引用变量可以指向子类对象。
多态的前提是必须有子父类关系或者类实现接口关系,否则无法完成多态。
在使用多态后的父类引用变量调用方法时,会调用子类重写后的方法。
子类对象指向父类引用变量;
多态格式:
父类类型 变量名 = new 子类类型();
变量名.方法名();
//普通类继承普通类,格式:父类类型 变量名=new 子类类型();
//普通类继承抽象类,格式:抽象类类型 变量名=new 普通类类型();
//普通类实现接口,格式:接口类型 变量名=new 普通类类型();
//在多态中成员变量的特点:
//编译时期看父类是否有该成员变量,有,编译成功,没有编译失败
//运行时期的结果走父类的成员变量
f.method();
//在多态中成员方法的特点:
//编译时期看父类是否有该成员方法,有,编译成功,没有,编译失败
//运行时期的结果是调用子类重写后的方法(前提是父类必须有该方法),如果子类没有重写,则调用父类成员方法;
//子类中独有的成员变量和成员方法,通过父类引用是不可以调用的;
注意事项:
① 同一个父类的方法会被不同的子类重写,在调用方法时,调用的为各个子类重写后的方法;
多态成员变量特点:编译时期和运行都参考=号左边(编译运行看左边);
多态成员方法特点:编译时期看左边,运行时期看右边(编译时期如果父类没有这个方法,则会编译报错,运行时期会执行子类重写后的方法,如果子类没有重写,则调用父类方法,抽象类和接口除外);
instanceof关键字:
可以通过instanceof关键字来判断某个对象是否属于某种数据类型;
多态转型:
① 向上转型:当有子类对象赋值给一个父类引用时,便是向上转型,多态本身就是向上转型的过程;
使用格式:
父类类型 变量名 = new 子类类型();
② 向下转型:一个已经向上转型的子类对象可以使用强制类型转换的格式,将父类引用转为子类引用,这个过程是向下转型。如果是直接创建父类对象,是无法向下转型的!
使用格式:
子类类型 变量名 = (子类类型) 父类类型引用变量;
通过向下转型,就可以调用子类独有的成员变量和方法;
double转int时,会出现损失精度的问题,但是在多态中向下转型,只要不转错就不会出BUG;
只有在子类对象交给父类引用时,才可以将父类引用向下转型;也就是说向下转型的前提是多态;
当父类的引用指向子类对象时,就发生了向上转型,即把子类类型对象转成了父类类型;向上转型的好处是隐藏了子类类型,提高了代码的扩展性;
但向上转型也有弊端,只能使用父类共性的内容,而无法使用子类特有功能,功能有限制;
向上转型或者向下转型,谨记使用instanceof关键字,避免ClassCastException!
总结:
① 什么时候使用向上转型:
当不需要面对子类类型时,通过提高扩展性,或者使用父类的功能就能完成相应的操作,这时就可以使用向上转型;
② 什么时候使用向下转型
当要使用子类特有功能时,就需要使用向下转型;
③ 向下转型的好处:可以使用子类特有功能;
弊端是:需要面对具体的子类对象;在向下转型时容易发生ClassCastException类型转换异常,需要使用instanceof关键字做判断;
String str=new String(“中国”);
等同于:
String str=”中国”;
面向对象总结:
① 封装:把对象的属性与方法的实现细节隐藏,仅对外提供一些公共的访问方式;
② 继承:子类会自动拥有父类所有可继承的属性和方法;
③ 多态:配合继承与方法重写提高了代码的复用性与扩展性;如果没有方法重写,则多态同样没有意义;子类对象指向父类引用……
构造方法/构造器/构造函数:
需要在创建对象的同时明确对象的属性值;
构造方法的格式:
修饰符 构造方法名(参数列表){
}
注意:
① 一个类中可以有无数个构造方法;
② 构造方法名与类名一致;
③ 没有返回值类型,也不需要写返回值(用于构造对象的方法,对象构建完,方法就执行结束);
建议:在创建有参构造的同时为类提供一个空参构造
一个对象只能调用一次构造方法;
构造方法内存图:
Main方法进栈,new Person()进堆,先给String name和int age分配初值,null和0,然后分配地址为0x001,然后找到Person的有参构造方法,进栈,传给有参构造的参数列表传值name=”张三”,age=18,然后给堆内的对象的成员变量赋值,然后把地址交给peo这个引用,构造方法执行完成,弹栈,然后打印名字和年龄;
类中默认有一个空参构造;
编译器在编译.java文件时,如果发现类中没有构造方法,会自动给你添加一个空参构造;
但是如果你的类中只要有一个构造方法,编译器将不会再给你添加默认的构造方法;
构造方法的细节:
① 一个类中可以有多个构造方法,多个构造方法是以重载的形式存在的;
② 构造方法是可以被private修饰的,使其他程序无法创建该类的对象;
构造方法和一般方法的区别:
① 构造方法在创建对象时执行,且只能被执行一次
② 一般方法在对象调用时执行,可以执行无数次
有了构造方法之后可以对对象的属性进行初始化,那么还需要对应的SET和GET方法吗?
需要相应的SET和GET方法,因为对象在创建之后需要修改和访问相应的属性值时,在这时只能通过SET或者GET方法来操作;
This作用:
① 区分成员变量和局部变量同名问题;
构造方法调用格式:
this(参数列表);
this关键字必须放在构造方法的第一行;
内存图解:
Main方法进栈,new Person在堆中开辟空间,String name和int age附初值,null和0,分配地址0x001,调用空参构造,构造方法进栈,发现又调用了一个构造方法this(“李四”,19);
有参构造进栈,之后将堆中初值改掉null改为李四,0改为19,有参构造弹栈,无参构造弹栈,将地址赋值给peo引用变量;
成员方法中形参是一个父类类型,传入的实际是一个子类类型,这是一个多态
成员方法中返回值类型是一个父类类型,实际返回是一个子类类型,这也是一种多态,不只是子类对象交给父类引用Person peo=new Student();才叫多态;