面向对象(中)
面向对象特征之二:继承性(extends)
继承性的好处
- 1、减少代码的冗余
- 2、便于功能的扩展
- 3、为之后的多态性的使用,提供了前提
继承性的格式:class A extends B{}
- A:子类、派生类、subclass
- B:父类、超类、superclass
体现
- 1、一旦子类继承父类B以后,子类A中就获取了父类B中声明的所有的属性和方法
特别的
,父类中声明为private的属性或方法,子类继承父类后,仍然认为获取了父类中私有的结构。只有因为封装性(private)的影响:使得子类不能直接调用父类的结构而已 - 2、子类继承父类以后,还可以声明自己特有的属性或方法:实现功能的拓展。
子类和父类的关系,不同于子集和集合的关系
exteds:扩展
Java中关于继承性的规定
- 1、一个类可以被多个子类继承
- 2、Java中类的单继承性:一个类只能有一个父类
- 3、多层继承
- 子父类是相对的概念
- 子类
直接继承
的父类称为直接父类
。间接继承
的父类称为间接父类
子类继承父类以后,就获取了直接父类和所有间接父类的属性和方法
说明
- 1、如果没有显示的声明一个类的父类,则此类继承于java.lang.Object类
- 2、所有的java类(除java.lang.Object类之外)都直接或间接的继承于java.lang.Object类
- 3、意味着,所有的java类具有java.lang.Object类声明的功能
方法的重写(override/overwrite)
重写
子类继承父类以后,可以对父类中同名参数的方法,进行覆盖
应用
重写以后,当创建子类对象以后,通过子类对象调用子父类中的同名同参数的方法时,实际执行的是子类重写父类的方法
重写的规定
方法的声明:权限修饰符 返回值类型 方法名(形参列表) throws 异常的类型{//方法体}
- 1、子类重写的方法的方法名和形参列表与父类被重写的方法的
方法名和形参列表相同
- 2、子类重写的方法的权限修饰符
不小于
被重写的方法的权限修饰符特殊情况
:子类不能重写父类中声明为private权限的方法
- 返回值类型
- 父类被重写的方法的
返回值类型是void
,则子类重写的方法的返回值类型只能是void
- 父类被重写的方法的返回值类型是
A类型
,则子类重写的方法的返回值类型可以是A类或A类的子类
- 父类被重写的方法的返回值类型是
基本数据类型
,则子类重写的方法的返回值类型也必须是相同的基本数据类型
- 父类被重写的方法的
- 子类重写的方法抛出的异常类型
不大于
父类被重写的方法抛出的异常类型
子类和父类中的同名参数的方法要么都声明为非static的(考虑重写),要么都声明为static的(不是重写)
super关键字
- 1、super理解为:调用父类中被重写的方法、属性
- 2、super可以用来调用:属性、方法、构造器
super的使用:调用属性和方法
- 1、可以在子类的方法或构造器中。通过使用"super.属性"或“super.方法”的方式,显示的调用父类中声明的属性或方法。但是,通常情况下,我们习惯省略“super.”
- 2、
特殊情况(属性)
:当子类和父类中定义了同名的属性时,我们要想在子类中调用父类中声明的属性,则必须显示的使用“super.属性名”,表明调用的是父类中声明的属性 - 3、
特殊情况(方法)
:当子类重写了父类中的方法以后,我们想要在子类的方法中调用父类中被重写的方法时,则必须显示的使用“super.方法”的方式,表明调用的时父类中被重写的方法
super的使用:调用构造器
- 1、我们可以在子类的构造器中显示的使用“super(形参列表)”的方式,调用父类中声明的指定的构造器
- 2、“super(形参列表)”的使用,必须声明在子类构造器的首行
- 3、在类的构造器中,针对“this(形参列表)”或“super(形参列表)”只能二选一,不能同时出现
- 4、在构造器的首行,没有显示的声明“this(形参列表)”或“super(形参列表)”,则默认调用的时父类中空参的构造器:super()
- 5、在类的多个构造器中,至少有一个类的构造器中使用了“super(形参列表)”,调用父类中的构造器
子类实例化的全过程
- 1、从结果上来看:(继承性)
- 子类继承父类以后,就获取了父类中声明的属性或方法
- 创建子类的对象,在对空间中,就会加载所有父类中声明的属性
- 2、从过程上来看
- 当我们通过子类的构造器创建子类对象时,我们一定会直接或间接的调用其父类的构造器,进而调用父类中的构造器,直到调用了java.lang.Object类中空参的构造器位置。正因为加载过所有的父类的结构,所以才可以看到内存中有父类中的结构,子类对象才可以考虑进行调用
虽然创建子类对象时,调用了父类的构造器,但是至始至终就创建了一个对象,即为new的子类对象
面向对象特征之三:多态性
- 1、可以理解为一个事物的多种形态
- 2、对象的多态性:父类的引用指向子类的对象(子类的对象赋给父类的引用)
- 3、多态的使用:虚拟方法调用
- 有了对象的多态性以后,编译期,只能调用父类中声明的方法,但在运行期,实际执行的是子类重写父类的方法
总结一句话:编译,看左边;运行看右边
- 有了对象的多态性以后,编译期,只能调用父类中声明的方法,但在运行期,实际执行的是子类重写父类的方法
eg:Person p = new Man();
- 4、多态性的使用前提
- a、要有类的继承关系
- b、要有方法的重写
- 5、对象的多态性,只适用于方法,不适用于属性
- 1、若子类重写了父类方法,就意味着子类里定义的方法彻底覆盖了父类里的同名方法,系统将不可能把父类里的方法转移到子类中:编译看右边,运行看左边
- 2、对于实例变量则不存在这样的现象,即使子类里定义了与父类完全相同的实例变量,这个实例变量依然不可能覆盖父类中定义的实例变量:编译运行都看左边
向下转型
instanceof关键字
a instanceof A:判断对象a是否是类A的实例。如果是,返回true,如果不是,返回false
使用情景:为了避免在向下转型时出现ClassCastException的异常,在向下转型之前,先进行instanceof的判断,一旦返回true,就进行向下转型。如果返回false,不进行向下转型
如果a instanceof A返回true,则a instanceof B也返回true。其中,类B是类A的父类
Object类
java.lang.Object类
- 1、Object类是所有Java类的根父类
- 2、如果在类的声明中未使用extends关键字指明其父类,则默认父类为java.lang.Object类
- 3、Object类中的功能(属性、方法)就具有通用性
- 属性:无
- 方法:
equals()、toString()
、getClass()、hashCode()、clone()、finalize()、wait()、notity()、notifyAll()
- 4、Ojbect类只声明了一个空参类型
‘==’和equals()区别
==运算符
- 1、可以使用在基本数据类型变量和引用数据类型变量中
- 2、如果比较的是
基本数据类型变量
:比较两个变量保存的数据值
是否相等。(不一定类型相同)
如果比较的是引用数据类型变量
:比较两个对象的地址
是否相同,即两个引用是否指向同一个对象实体
equals()方法
- 1、是一个方法,而非运算符
- 2、只能适用于引用数据类型
- 3、Object类中equals()的定义;
public boolean equals(Object obj){
return (this == obj);
}
说明:Object类中定义的equals()和 ‘==’的作用是相同
的:比较两个对象的地址值是否相同
,即两个引用是否指向同一个地址
- 4、像
String、Date、File、包装类
等都重写了Object类中的equals()方法。重写以后,比较的不是两个引用地址是否相同,而是比较两个对象的“实体内容”是否相同
。 - 5、通常情况下,我们自定义的类如果使用equals()的话,也通常是比较两个对象的“实体内容”是否相同。那么我们就需要对Object类中的equals()进行重写。
重写原则:比较两个对象的实体内容(即:name和age)是否相同
toString()的使用
- 1、当我们输出一个对象的引用时,实际上就是调用当前对象的toString()
- 2、Object类中toString()的定义
public String toString(){
return getClass().getName() + "@" + Integer.toHexString(hashCode());
}
- 3、像String、Date、File、包装类等都重写了Object类中的toString()方法
使得在调用对象的toString()时,返回“实体内容”信息 - 4、自定义类也可以重写toString()方法,当调用此方法时,返回对象的“实体内容”
单元测试
步骤
- 1、选中当前工程->右键选择:build path -> add libraries -> JUnit ->下一步
- 2、创建Java类,进行单元测试
此时的Java类要求:
- 此类是public的
- 此类提供公共的无参的构造器 - 3、此类中声明单元测试方法
此时的单元测试方法:
- 方法的权限是public,没有返回值,没有形参 - 4、此单元测试方法上需要声明注释:@Test,并在单元测试类中导入:import org.junit.Test;
- 5、声明好单元测试方法以后,就可以在方法体内测试相关的代码
- 6、写完代码以后,左键双击单元测试方法名,右键:run as -> JUnit
说明
- 1、如果执行结果没有任何异常:绿条
- 2、如果执行结果出现异常:红条
包装类的使用
八种基本数据类型定义相应的引用类型–包装类(封装类)
框起来那几个有一个共同的父类Number,在往上,Number的父类是Object类
基本类型、包装类与String类间的转换
-
基本数据类型—>包装类:调用包装类的构造器
- 自动装箱:基本数据类型—>包装类
int num2 = 10;
Integer in1 = num2;//自动装箱
boolean b1 = true;
Boolean b2 =b1;//自动装箱
- 包装类—>基本数据类型:调用包装类Xxx的xxxValue()
Integer in1 = new Integer(12);
int i1 = in1.intValue();
System.out.println(i1 + 1);
- 自动拆箱:包装类--->基本数据类型
int num3 = in1;//自动拆箱
- 基本数据类型、包装类—>String类型
- String类型—>基本数据类型、包装类:调用包装类的parseXxx(String s)方法