封装、继承与多态

163 阅读6分钟

封装介绍

封装(encapsulation)就是把抽象出的数据[属性]和对数据的操作[方法]封装到一起,数据被保护在内部,程序的其他部分只有通过被授权的操作[方法],才能对数据进行操作。

封装的好处和理解

1)隐藏实现细节:方法(连接数据库)<–调用(传入参数。。)

2)可以对数据进行验证,保证安全合理

封装实现的步骤

1)将属性进行私有化private【不能直接修改属性】

2)提供一个公共的(public)set方法,用于对属性进行判断并赋值

public void setXxx(类型 参数名){//Xxx 表示某个属性、

//加入数据验证的业务逻辑

属性 = 参数名;

}

3)提供一个公共的get方法,用于获取属性的值

public XX getXxx(){//权限判断

return xx;

}

继承

继承原理图

继承可以解决代码复用,让我们编程更加靠近人类思维,当多个类存在相同属性(变量)和方法时,可以从这些类中抽象出父类,在父类中定义这些相同的属性和方法,所有子类不需要重新定义这些属性和方法,只需要通过extends来声明继承父类即可。

继承基本语法

class 子类 extends 父类 {

}

1)子类就会自动拥有父类定义的属性和方法

2)父类又叫超类,基类

3)子类又叫派生类

继承带来的便利

1)代码的复用性提高

2)代码的扩展性和维护性提高

继承的使用细节

1.子类继承了所有的属性和方法,非私有的属性和方法可以在子类直接访问,但是私有属性和方法不能在子类直接访问,要通过公共的方法进行访问(get之类)。

->默认(default)子类不能访问但是子类父类通常在一个包里面所以可以访问到

子类与父类在同一个包下,子类可以访问父类默认修饰符修饰的属性

子类与父类不在同一个包下,子类不可以访问父类默认修饰符修饰的属性

2.子类必须调用父类构造器,完成父类的初始化

3.当创建子类对象的时候,不管使用子类的哪个构造器,默认情况下总会去调用父类的无参构造器,如果父类没有提供无参构造器,则必须在子类的构造器中用super(参数,参数)去指定使用父类的哪个构造器完成对父类的初始化工作,否则编译报错。

4.如果希望指定去调用父类的某个构造器,则显式的调用一下:super(参数列表)

5.super在使用时,必须放在构造器第一行(super只能在构造器中使用)

6.super()和this()【this()用于在构造器中调用另外一个构造器】都只能在构造器第一行,因此两个方法不能共存在一个构造器。

7.java所有类都是Object类的子类,Object是所有类的基类。

8.父类构造器的调用不限于直接父类,将一直往上追溯直到Object类(顶级父类)。

9.子类最多只能继承一个父类(指直接继承),即java中是单继承机制,思考如何让A类继承B类和C类(A继承B。B继承C)

10.不能滥用继承,子类和父类之间必须满足is – a 的逻辑关系

继承的本质分析(重要)

当子类继承父类,创建子类对象时,内存中发生了什么?->当子类对象创建好后,建立查找的关系

当创建son对象时,初始化对象从object先初始化一级一级往下初始化最终到son

堆里面只有一个Son对象,son对象里面有GrandPa Father 的所有属性和方法

查找属性规则

(1)首先看子类是否有这个属性

(2)如果子类有这个属性而且可以访问,则直接返回信息

(3)如果子类没有这个属性,就看父类有没有这个属性(如果父类有该属性,并且可以访问,就返回信息)

(4)如果父类没有就按照(3)的规则,继续往上级父类,直到Object

(5)如果父类的某个属性(比如age)为私有的,就算父类的父类(GrandPa)有age属性也无法访问会报错

super关键字

super代表父类的引用,用于访问父类的属性、方法、构造器

~基本语法

1.访问父类属性,但不能访问父类的private属性—>super.属性名;

子类与父类在同一个包下,子类可以访问父类默认修饰符修饰的属性

子类与父类不在同一个包下,子类不可以访问父类默认修饰符修饰的属性

2.访问父类的方法,不能访问父类的private方法—>super.方法名(参数列表);

3.访问父类构造器:super(参数列表);只能放在构造器的第一句!

super使用细节

1.调用父类的构造器的好处(分工明确,父类属性由父类初始化,子类属性由子类初始化)

2.当子类中有和父类中的成员(属性和方法)重名时,为了访问父类成员,必须通过super。如果没有重名使用super、this、直接访问效果一样。

属性的规则同理

3.super的访问不限于直接父类,如果爷爷类和本类中有同名的成员,也可以使用super去访问爷爷类的成员;如果多个基类(上级类)中都有同名的成员。使用super访问遵循就近原则。A-B-C,当然也需要遵守访问权限的相关规则。(如果需要跳过父类访问爷爷类,则可以在爷爷类中定义相关的方法返回即可)

方法重写

基本介绍

方法覆盖(重写)就是子类有一个方法,和父类的某个方法的名称、返回类型、参数一样,那么我们就说子类的这个方法覆盖了父类的方法。

方法重写的注意事项和使用细节

1.子类的方法的形参列表方法名称,要和父类方法的形参列表方法名称完全一样。

2.子类方法的返回类型和父类方法返回类型一样,或者是父类返回类型的子类,比如父类返回类型是Object,子类方法返回类型是String

3.子类方法不能缩小父类方法的访问权限 public > protected > 默认 > private

面向对象编程–多态

对象的多态

(1)一个对象的编译类型和运行类型可以不一致

(2)编译类型在定义对象时,就确定了不可以改变

(3)运行类型可以变化

(4)编译类型看定义时 = 号的左边,运行类型看 = 号的右边

向下转型

√多态的前提:两个对象(类)存在继承关系

√多态的向上转型

1)本质:父类的引用指向子类的对象 ———-> Animal animal = new Dog();

2)语法: 父类 引用名 – new 子类类型();

  1. 特点:编译类型看左边,运行类型看右边。

可以调用父类中所有的成员方法(需要遵循访问权限),

不能调用子类特有成员;

最终运行效果看子类的具体实现!

向下转型

1)语法 : 子类类型 引用名 = (子类类型) 父类引用;

2)只能强转父类的引用,不能强转父类的对象

2)要求父类的引用必须指向的是当前目标类型的对象

4)当向下转型后,可以调用子类类型中所有的成员

Dog dog = (Dog) animal; 是错误的 因为animal 指向的是一个cat对象无法将cat对象变成Dog对象

属性没有重写之说!属性的值看编译类型

instanceOf 比较操作符,用于判断对象的运行类型是否为XX类型或者XX类型的子类型

java的动态绑定机制非常重要

1:当调用对象方法的时候,该方法会和该对象的内存地址/运行类型绑定

2:当调用对象属性的时候,没有动态绑定机制,哪里声明,哪里使用

开启掘金成长之旅!这是我参与「掘金日新计划 · 2 月更文挑战」的第 18 天,点击查看活动详情