5 面向对象的特征
封装
继承
多态
5.1 封装
隐藏对象的属性和实现细节,仅对外提供公共访问方式 好处:便于使用
提高重用性
提高安全性
例如:函数(最小的封装体)、类、包、框架。
5.2 继承
1、继承的特点
提高了代码的复用性。
类和类之间之间产生了关系,从而有了多态的特性。
java只支持单继承,以为多继承容易带来安全隐患:当多个父类中定义了相同功能,但功能内容不同时,子类对象不确定运行哪一个。但是java保留了这种机制,用接口来完成多实现。
java支持多层继承。
子类和父类中有相同的变量,子类会覆盖父类的变量。
当子类中出现和父类一模一样的函数,当子类对象调用该函数时,会运行子类函数的内容,这就是函数的重写,也叫做覆盖。子类覆盖父类,必须保证子类权限大于等于父类权限才可以覆盖。
静态只能覆盖静态
若要在子类中访问父类中的同名变量或方法时,需要使用关键字super.变量名/方法
在对子类对象初始化时,父类的构造函数也会运行,那是因为子类的构造函数默认第一行有一条隐式得得语句super();super()会访问父类的无参构造函数,而且子类中的构造函数默认第一行都是super();
2、为什么子类一定要访问父类的构造函数?
因为父类中的数据子类可以直接获取,所以子类对象在建立时,需要先查看父类是如何对这些数据进行初始化的。所以子类在对象初始化时,先要访问一下父类中的构造函数。
如果父类中没有无参构造函数,需要在子类的构造函数中手动定义super语句。
super语句一定定义在子类构造函数的第一行。
5.3 多态
可以理解为事物存在的多种体现形态
1、多态的体现
父类的引用也可以接受自己的子类对象
2、多态的好处
提高了程序额扩展性
3、多态的前提
类与类之间是继承或实现的关系
存在方法的覆盖
4、多态的弊端
只能使用父类的引用访问父类中的成员
5、在多态中成员函数的调用:编译看左边,运行看右边
6、在多态中成员变量额特点:无论编译和运行都参考左边
5.4 final关键字
final是一个修饰符,可以修饰类、函数、变量。
被final修饰的类不可以被继承,为了避免被继承,被子类复写功能
被final修饰的方法不可以被复写
被final修饰的变量是一个常量,只能赋值一次,既可以修饰成员变量,也可以修饰局部变量。常量的命名规范是所有字母都大写,如果有多个单词组成,单词间通过下划线_连接
内部类定义在类中的局部位置上时,只能访问该局部被final修饰的局部变量
5.5 抽象类
抽象类中既可以有抽象方法,也可以有非抽象方法
抽象方法一定定义在抽象类中
抽象方法和抽象类都必须被abstract关键字修饰
抽象类不可以用new创建对象,因为调用抽象方法没意义
抽象类中的抽象方法要被调用,必须由子类复写所有的抽象方法后,建立子类对象调用。如果子类只覆盖了部分抽象方法,那么该子类还是一个抽象类。
5.6 接口
使用interface关键字声明一个接口
特点:
-
接口中的属性都是静态常量,默认public static final
接口中的方法都是抽象方法,默认public abstract
接口不可以创建对象,因为有抽象方法
接口需要被子类实现,子类对接口中的方法全都覆盖后,子类才可以实例化,否则子类是一个抽象类
接口可以被类多实现,而且接口之间可以有多继承
5.7 内部类
1、访问规则:
内部类可以直接访问外部类中的成员,包括私有
外部类要访问内部类,必须建立内部类对象
2、访问格式:
外部类名.内部类名 变量名 = 外部类对象.内部类对象
Outer.Inner inner = new Outer().new Inner();
当内部类在成员位置上,就可以被成员修饰符所修饰,比如
private:将内部类在外部类中进行封装
static:内部类就具备static的特性。这时内部类只能直接访问外部类中的static成员
3、注意:
当内部类中定义了静态成员,该内部类必须是static的
当外部类中的静态方法访问内部类时,内部类也必须是static的
内部类定义在局部时,不可以被成员修饰符修饰
可以直接访问外部类中的成员,但是不可以访问他所在的局部中的变量,只能访问被final修饰的局部变量匿名内部类
匿名内部类其实就是内部类的简写格式
4、定义匿名内部类的前提
内部类必须是继承一个类或者实现接口
5、匿名内部类的格式
new 父类或者接口(){定义子类的内容}
其实匿名内部类就是一个匿名子类对象
5.8 异常
异常就是程序在运行时出现的不正常情况,是对问题的描述,将问题进行封装
1、问题的划分:
一种是严重的问题,另一种是非严重的问题
对于严重问题,通过Error类进行描述。一般不编写针对性的代码对其进行处理
对于非严重问题,通过Exception类进行描述,使用针对性的方式进行处理
2、Error和Exception的父类是Throwable
Throwable
|--Error
|--Exception
3、异常处理
try{
需要被检测的代码;
}catch(异常类 变量){
处理异常的代码(处理方式);
}finally{
不管是否出错,一定会执行的语句;
(但是使用System.exit(0)时,不会执行)
}
finally一定会执行的代码,通常用于关闭资源
catch是用于处理异常,如果没有catch就代表异常没有被处理过。如果是检测时异常,就必须声明出去
4、异常特点
异常体系中的所有类以及建立的对象都具有可抛性,可以被throw和throws关键字所操作
5、throws Exception(放在方法的后面)
通过throws关键字声明了该功能可能会出现问题
声明的异常需要用try/catch处理,声明几个异常,就对应几个catch块
6、对捕获到的异常对象进行常见操作
String getMessage();获取异常信息
String toString();异常名称:异常信息
void printStackTrace();异常名称,异常信息,异常出现的位置。也是jvm默认的异常处理机制
7、throws和throw的区别
throws使用在函数上
throw使用在函数内
当函数内容有throw抛出异常对象,并未进行try处理,必须在函数上声明(RuntimeException除外)
如果函数声明了异常,调用者需要进行处理,处理方法有throws和try
throws后面跟的是异常类,可以跟多个,用逗号隔开
throw后面跟的是异常对象
8、Exception中有一个特殊的子类异常RuntimeException运行时异常
如果函数内容抛出该异常,函数上可以不用声明
如果函数上声明了该异常,调用者可以不用进行处理
9、异常的分类:
(1) 编译时被检测的异常
该异常在编译时,如果没有处理,则编译失败
(2)编译时不被检测额异常(运行时异常,RuntimeException及其子类)
在编译时,不需要处理,编译器不检查
该异常的发生,建议不处理,让程序停止
10、异常在子父类覆盖中的体现
子类在覆盖父类时,如果父类的方法抛出异常,那么子类的覆盖方法,只能抛出父类的异常或者该异常的子类
如果父类方法抛出多个异常,那么子类覆盖方法时,只能抛出父类异常额子集
如果父类或者接口额方法中没有异常抛出,那么子类覆盖方法时,也不可以抛出异常。若真发生异常,就必须要进行try/catch处理。
11、自定义异常
定义类继承Exception或者RuntimeException,为了让该自定义累具备可抛性,让该类具备操作异常的共性方法
5.9 包(package)
对类文件分类管理
给类提供多层命名空间
卸载程序文件的第一行
类名的全称是 包名.类名
包也是一种封装形式
1、包与包之间进行访问
被访问的包中的类以及类中的成员,需要public修饰
不同包中的子类还可以直接访问父类中被protected权限修饰的成员
包与包之间可以使用的权限只有两种,public、protected
2、权限范围
public protected default private
同一个类中 ok ok ok ok
同一个包中 ok ok ok
子类 ok ok
不同包中 ok
3、import
为了简化类名的书写,使用import关键字
import导入的是包中的类