浅谈java 面对对象(抽象 继承 接口 多态)
|
什么是继承? 多个类中存在相同属性和行为时,将这些内容抽取到单独一个类中,那么多个类无需再定义这些属性和行为,只要继承那个类即可。 多个类可以称为子类,单独这个类称为父类、超类或者基类。 子类可以直接访问父类中的非私有的属性和行为。 通过 extends 关键字让类与类之间产生继承关系。 class SubDemo extends Demo{} //SubDemo是子类,Demo是父类 继承有什么好处? •提高代码的复用性。 •让类与类之间产生了关系,是多态的前提。 继承的特点 1、Java只支持单继承,不支持多继承。
//一个类只能有一个父类,不可以有多个父类。
class SubDemo extends Demo{} //ok
class SubDemo extends Demo1,Demo2...//error
2.Java支持多层(重)继承(继承体系)。
class A{}
class B extends A{}
class C extends B{}
使用继承时的注意事项 •如果类之间存在着:is a 的关系,就可以考虑使用继承。 •不要为了继承部分功能,而去使用继承。 super和this有什么区别? super是一个关键字,代表父类的存储空间标识。(可以理解为父亲的引用) super和this的用法相似。 this代表对象的引用(谁调用就代表谁); 使用场景 •当子父类出现同名成员时,可以用super进行区分; 区别 1.成员变量 this.变量 -- 本类的 2.构造方法 this(...) -- 本类的 3.成员方法 this.方法名() -- 本类的 super();和this();都是在构造函数的第一行,不能同时出现。 方法的重写(覆盖) 子类中出现与父类一模一样的方法时(除了权限修饰符,权限修饰符大于等于不包括private,返回值类型,方法名和参数列表相同),会出现覆盖操作,也称为重写或者复写。 父类私有方法,子类看不到,因此父类私有方法的重写也就无从谈起。 覆盖注意事项: •覆盖时,子类方法权限一定要大于等于父类方法权限; •静态只能覆盖静态。 覆盖的使用场景: 当子类需要父类的功能,而功能主体子类有自己特有内容时,可以复写父类中的方法,这样,既沿袭了父类的功能,又定义了子类特有的内容。 方法重写和重载有什么区别? 方法的重写用在子类方法与父类方法一模一样时,除权限修饰符,返回值类型,方法名和参数列表都是相同的。 重载用在同一个类中各方法方法名相同,参数列表不同(与返回值类型没有关系)的情况。 子父类中构造方法的用法: 1.子类的初始化过程中,首先回去执行父类的初始化动作。因为子类的构造方法中默认有一个super()。子类要使用父类的成员变量,这个初始化,必须在子类初始化之前完成。所以,子类的初始化过程中,会先执行父类的初始化。 2.如果父类没有无参构造方法 •使用super调用父类的带参构造。推荐方式。 •使用this调用本身的其他构造。 静态代码块、构造代码块,构造方法的执行顺序: 父类静态代码块→子类静态代码块→父类构造代码块→父类构造方法→子类构造代码块→子类构造方法 final关键字 final是一个关键字,可以用于修饰类,成员变量,成员方法。 特点: 它修饰的类不能被继承。 它修饰的成员变量是一个常量。 它修饰的成员方法是不能被子类重写的。 final修饰的常量定义一般都有书写规范,被final修饰的常量名称,所有字母都大写。 final修饰成员变量,必须初始化,初始化有两种 显示初始化; 构造方法初始化。 final和private的区别: final修饰的类可以访问; final修饰的方法不可以被子类重写; final修饰的变量只能在显示初始化或者构造函数初始化的时候赋值一次,以后不允许更改;
概念: 对象在不同时刻表现出来的不同状态。 多态的前提: 要有继承或者实现关系。 要有方法的重写。 要有父类引用指向子类对象。 程序中的体现: 父类或者接口的引用指向或者接收自己的子类对象。 好处和作用: 多态的存在提高了程序的扩展性和后期可维护性。 弊端: 父类调用的时候只能调用父类里的方法,不能调用子类的特有方法,因为你并不清楚将来会有什么样的子类继承你。 多态的成员特点: 成员变量:编译时期:看引用型变量所属的类中是否有所调用的变量; 运行时期:也是看引用型变量所属的类是否有调用的变量。 成员变量无论编译还是运行都看引用型变量所属的类,简单记成员变量,编译和运行都看等号左边。 成员方法:编译时期:要查看引用变量所属的类中是否有所调用的成员; 运行时期:要查看对象所属的类中是否有所调用的成员。如果父子出现同名的方法,会运行子类中的方法,因为方法有覆盖的特性。 编译看左边运行看右边。 静态方法:编译时期:看的引用型变量所属的类中是否有所调用的变量; 运行时期:也是看引用型变量所属的类是否有调用的变量。 编译和运行都看等号左边。 一定不能够将父类的对象转换成子类类型! 父类的引用指向子类对象,该引用可以被提升,也可以被强制转换。 多态自始至终都是子类对象在变化!
//多态向下转型和向上转型的例子,多态转型解决了多态中父类引用不能使用子类特有成员的弊端。
class PolymorphicTest2 {
public static void main(String[] args) {
Phone p1 = new Nokia(); //向上转型,类型提升
Nokia no = (Nokia)p1; //向下转型,强制将父类的引用转换成子类类型,不能将Nokia类型转成Moto或Nexus类型
no.print(); //输出结果为Phone---null---0,因为继承了父类的方法
Phone p2 = new Moto();
Moto m = (Moto)p2;
m.print(); //输出结果为Moto---yellow---1599,方法重写,子类方法覆盖父类方法
Phone p3 = new Nexus();
Nexus ne = (Nexus)p3;
ne.print();
}
}
class Phone{
String color;
int price;
public void print(){
System.out.println("Phone---" + color + "---" + price );
}
}
class Nokia extends Phone{
String color = "red";
int price = 1009;
//public void print(){
// System.out.println("Nokia---" + color + "---" + price);
/ |
