面向对象的三大特征
1. 封装性(能不能调用)
- 封装性的设计思想:隐藏对象内部的复杂性,只对外公开简单的接口,便于外界调用,从而提高系统的
可扩展性、可维护性
。通俗的说:把该隐藏的隐藏起来,该暴露的暴露出来
- Java中封装性的体现
- 将类的属性xxx私有化(private),然后提供公共(public)的方法来获取(getXxx)和设置(setXxx)属性的值
- 不对外暴露私有的方法
- 单例模式(构造器)
- 如果不希望类在包外被调用,可以将类设置为缺省
- 封装性的体现需要权限修饰符来配合
权限修饰符 类内部 同个包 不同包的子类 同一个工程 private √ 缺省(default) √ √ protected √ √ √ public √ √ √ √
注意:对于class的权限修饰只可以用public和default(缺省)
2. 继承性(能不能获取)
- 继承性的好处
- 减少了代码的冗余 ,提高了代码的复用性
- 便于功能的扩展
- 为之后多态性的使用,提供了前提
- 继承的体现:一旦子类A继承父类B以后,子类A中就获取了父类B中声明的所有属性和方法。
- 父类中声明为private的属性或方法,子类仍然有继承,但因为封装性的影响,使得子类不能直接调用父类中私有的结构而已。
- 如果我们没有显示的声明一个类的父类的话,则此类继承于java.lang.Object类
- 方法的重写
- 子类重写的方法的
方法名和形参列表
与父类被重写的方法的方法名和形参列表相同- 子类重写的方法的
权限修饰符
不小于父类被重写的方法的权限修饰符
特殊情况:子类不能重写父类中声明为private权限的方法
返回值
类型
① 父类被重写的方法的返回值的类型时void,则子类中重写的方法的返回值也必须是void
②父类被重写的方法的返回值是基本数据类型(如double),则子类重写的方法的返回值也必须是相应的基本数据类型(如double)
③父类被重写的方法的返回值类型是A类型,则子类重写的方法的返回值类型可以是A类或A类的子类- 子类重写的方法
抛出的异常类型
不大于父类被重写的方法抛出的异常
注意:子类和父类中的同名同参数方法要么都声明为非static的(考虑重写),要么都声明为static的(不是重写)
- super关键字的使用
super调用属性和方法
①我们可以在子类的方法或构造器中,通过使用‘super.属性’或‘super.方法’的方式,显式的调用父类中声明的属性或方法。但是通常情况下,我们习惯省略‘super.’
②特殊情况下:当子类和父类定义了同名的属性时,我们想要在子类中调用父类声明的属性,则必须显式的使用‘super.属性’的方式,表明调用的是父类中声明的属性
③ 特殊情况下:当子类重写父类的方法后,我们想要在子类中调用父类被重写的方法时,则必须显式的使用‘super.方法’的方式,表明调用的是父类中被重写的方法。super调用构造器
①我们可以在子类的构造器中显示的使用‘super(形参列表)’的方式,调用父类中声明指定的构造器
②‘super(形参列表)’的使用,必须在子类构造器的首行
③在类的构造器中,针对于‘this(形参列表)’和‘super(形参列表)’只能二选一,不能同时出现
④在构造器的首行,没有显示的声明‘this(形参列表)’或‘super(形参列表)’,则默认调用父类中空参构造器,super()
⑤在类的多个构造器中,至少有一个构造器使用了‘super(形参列表)’
- 子类对象实例化的全过程
- 从结果上看:(继承性)子类继承父类以后,就获取了父类中声明的属性或方法。创建子类的对象,在堆空间中,就会加重所有父类中声明的属性。
- 从过程上看:当我们通过子类构造器创建子类的对象时,我们一定会直接或间接的调用其父类的构造器,进而调用父类的父类的构造器。。。直到调用了java.lang.Object类中的空参构造器为止。正因为加载过所有父类的构造器,才可以看到内存中有父类中的结构,子类对象才可以考虑进行调用
注意:虽然创建子类对象时,调用了父类的构造器,但是自始至终就创建了一个对象,即为new的子类对象。
3. 多态性(多态是运行时行为
,主要目的是通用性)
- 理解多态性:可以理解为一个事物的多种形态
- 对象的多态性:父类的引用指向子类的对象
- 多态的使用:虚拟方法的调用
有了对象的多态性以后,我们在编译期,只能调用父类中声明的方法,但在运行期,我们实际执行的是子类重写服了父类的方法。
总结:编译看左边;运行看右边(方法)
- 多态性使用的前提:①类的继承关系 ②方法的重写
- 对象的多态性,只适用于方法,不适用于属性(编译和运行都看左边)
- 虚拟方法调用(动态绑定):子类中重写了父类的方法,在多态情况下,将此时父类的方法称为虚拟方法,父类根据赋予给它的不同子类对象,
动态的调用属于子类的该方法
。这样的方法在编译期是无法确定的。
向下转型
:
- 为什么要向下转型:有了对象的多态性以后,内存中实际上时加载了子类特有的属性和方法的,但是由于变量声明为父类类型,导致编译时,只能调用父类中声明的属性和方法,子类特有的属性和方法不能调用。
- 解决方法:使用
instanceof
判断后向下转型