继承
1,引入和概述
A:继承概述
多个类中存在相同属性和行为时,将这些内容抽取到单独一个类中,那么多个类无需再定义这些属性和行为,只要继承那个类即可。
B:继承格式
通过extends关键字可以实现类与类的继承
class 子类名 extends 父类名 {}
单独的这个类称为父类,基类或者超类;这多个类可以称为子类或者派生类。
2, 好处和弊端
1,继承的好处
a:提高了代码的复用性
b:提高了代码的维护性
c:让类与类之间产生了关系,是多态的前提
2,继承的弊端
类的耦合性增强了。
开发的原则:高内聚,低耦合。
耦合:类与类的关系
内聚:就是自己完成某件事情的能力
需求:
学生类和老师类。
定义两个功能(吃饭,睡觉)
3, 类的继承特点
Java中类的继承特点
a:Java只支持单继承,不支持多继承。(一个子类只能继承一个父类)
有些语言是支持多继承,格式:extends 类1,类2,...
b:Java支持多层继承(继承体系)
4,继承的注意事项
继承的注意事项
a:子类只能继承父类所有非私有的成员(成员方法和成员变量)
b:子类不能继承父类的构造方法(构造方法不参与继承),但是可以通过super(待会儿讲)关键字去访问父类构造方法。
c:不要为了部分功能而去继承
什么时候使用继承
继承其实体现的是一种关系:"is a" .
采用假设法。
如果有两个类A,B。只有他们符合A是B的一种,或者B是A的一种,就可以考虑使用继承
5,继承中成员变量的关系
A:子类中的成员变量和父类中的成员变量名称不一样
B:子类中的成员变量和父类中的成员变量名称一样
在子类中访问一个变量的查找顺序("就近原则")
a: 在子类的方法的局部范围找,有就使用
b: 在子类的成员范围找,有就使用
c: 在父类的成员范围找,有就使用
d: 如果还找不到,就报错
public class MyTest {
public static void main(String[] args) {
Son son = new Son();
son.show(20);
}
}
class Father{
int num=200;
}
class Son extends Father{
int num=600;
public void show(int num){
System.out.println(num);
System.out.println(this.num);
System.out.println(super.num);
}
}
6, this和super的区别
说说this和super的区别
this 代表的是本类对象的引用
super代表的是父类存储空间的标识(可以理解成父类的引用,可以操作父类的成员)
C:this和super的使用
a:调用成员变量
this.成员变量 调用本类的成员变量
super.成员变量 调用父类的成员变量
b:调用构造方法
this(...) 调用本类的构造方法
super(...) 调用父类的构造方法
c:调用成员方法
this.成员方法 调用本类的成员方法
super.成员方法 调用父类的成员方法
7,继承中构造方法的关系
子类中所有的构造方法默认都会访问父类中空参数的构造方法
为什么呢?
因为子类会继承父类中的数据,可能还会使用父类的数据。
所以,子类初始化之前,一定要先完成父类数据的初始化。
其实:
每一个构造方法的第一条语句默认都是:super()
在这里简单的提一句,Object类。否则有人就会针对父类的构造方法有疑问。Object在没有父类了。
public class MyTest extends Object{
public static void main(String[] args) {
Zi zi = new Zi();
}
}
class Fu extends Object{
int fuNum=100;
public Fu() {
super();
System.out.println("父类的构造方法调用了");
}
}
class Zi extends Fu{
int ziNum=200;
public Zi() {
super();
System.out.println("子类的构造方法调用了");
}
}
8, 继承中构造方法的注意事项
父类没有无参构造方法,子类怎么办?
a: 在父类中添加一个无参的构造方法
b:子类通过super去显示调用父类其他的带参的构造方法
c:子类通过this去调用本类的其他构造方法
本类其他构造也必须首先访问了父类构造
B:注意事项
super(…)或者this(….)必须出现在构造方法的第一条语句上
this()与super()不能同时出现在一个构造方法中
public class MyTest {
public static void main(String[] args) {
Zi zi = new Zi();
Zi zi1 = new Zi(20);
}
}
class Fu extends Object{
public Fu(int num) {
super();
System.out.println("父类的有参构造");
}
}
class Zi extends Fu{
public Zi() {
super(10);
System.out.println("子类 构造执行了");
}
public Zi(int num){
this();
System.out.println("子类有参构造执行了");
}
}
9, 继承中成员方法的关系
a:当子类的方法名和父类的方法名不一样的时候
b:当子类的方法名和父类的方法名一样的时候
通过子类调用方法:
a: 先查找子类中有没有该方法,如果有就使用
b:在看父类中有没有该方法,有就使用
c: 如果没有就报错
10, 方法重写
A:什么是方法重写
子类中出现了和父类中一模一样的方法声明(方法名,参数列表,返回值类型),也被称为方法覆盖,方法复写。
B:方法重写的应用:
当子类需要父类的功能,而功能主体子类有自己特有内容时,可以重写父类中的方法。
这样,即沿袭了父类的功能,又定义了子类特有的内容。
方法重写的注意事项
A:方法重写注意事项
a:父类中私有方法不能被重写
因为父类私有方法子类根本就无法继承
b:子类重写父类方法时,访问权限不能更低
最好就一致
c:父类静态方法,子类也必须通过静态方法进行重写 静态方法不参与重写
其实这个算不上方法重写,但是现象确实如此,至于为什么算不上方法重写,多态中我会讲解
子类重写父类方法的时候,最好声明一模一样。
11,final关键字
为什么会有final
由于继承中有一个方法重写的现象,而有时候我们不想让子类去重写父类的方法.这对这种情况java就给我们提供了一个关键字: final
final概述
final关键字是最终的意思,可以修饰类,变量,成员方法。
final修饰特点
修饰类: 被修饰类不能被继承
修饰方法: 被修饰的方法不能被重写
修饰变量: 被修饰的变量不能被重新赋值,因为这个量其实是一个常量
修饰局部变量:
基本类型,是值不能被改变
引用类型,是地址值不能被改变
多态
1, 概述及代码体现
A:多态概述
某一个事物,在不同时刻表现出来的不同状态。
举例: Cat c=new Cat();
Animal a=new Cat();
猫可以是猫的类型。猫 m = new 猫();
同时猫也是动物的一种,也可以把猫称为动物。动物 d = new 猫();
B:多态前提
a:要有继承关系。
b:要有方法重写。 其实没有也是可以的,但是如果没有这个就没有意义。
c:要有父类引用指向子类对象。
父 f = new 子();
2, 多态中的成员访问特点
A:多态中的成员访问特点
a:成员变量
编译看左边,运行看左边。
b:构造方法
创建子类对象的时候,会访问父类的构造方法,对父类的数据进行初始化。
c:成员方法
编译看左边,运行看右边。
d:静态方法
编译看左边,运行看左边。
(静态和类相关,算不上重写,所以,访问还是左边的)
*******结论:由于成员方法存在方法重写,所以它运行看右边(子类)
B:案例演示
多态中的成员访问特点
public class MyTest {
public static void main(String[] args) {
Fu fu=new Zi();
System.out.println(fu.num);
fu.show();
fu.function();
}
}
3, 多态的好处
多态的好处
a:提高了代码的维护性(继承保证)
b:提高了代码的扩展性(由多态保证)
4, 多态的弊端 及多态中的向上转型和向下转型
A:通过多态的弊端引出问题
不能使用子类特有的功能
B:解决问题
a:把父类的引用强制转换为子类的引用。(向下转型)
C:案例演示
详细讲解多态中向上转型和向下转型
public class MyTest {
public static void main(String[] args) {
Fu fu = new Zi();
fu.show();
Zi zi= (Zi) fu;
zi.method();
}
}
class Fu{
public void show(){
System.out.println("父类的show方法");
}
}
class Zi extends Fu{
@Override
public void show() {
System.out.println("我重写了父类的show方法");
}
public void method(){
System.out.println("这是子类特有的一个方法");
}
}
5, 多态中的内存图
6,多态的一道分析题
class A {
public void show() {
show2();
}
public void show2() {
System.out.println("我");
}
}
class B extends A {
public void show2() {
System.out.println("爱");
}
}
class C extends B {
public void show() {
super.show();
}
public void show2() {
System.out.println("你");
}
}
public class DuoTaiTest4 {
public static void main(String[] args) {
A a = new B();
a.show();
B b = new C();
b.show();
}
}
抽象类
1,概述及其特点
不同的动物吃的东西应该是不一样的,所以,我们不应该在动物类中给出具体体现,而是应该给出一个声明即可。
在Java中,一个没有方法体的方法应该定义为抽象方法,而类中如果有抽象方法,该类必须定义为抽象类。
B:抽象类特点
a:抽象类和抽象方法必须用abstract关键字修饰
抽象类格式: abstract class 类名 {}
抽象方法格式: public abstract void eat();
b:抽象类不一定有抽象方法,有抽象方法的类一定是抽象类
c:抽象类中可以有构造方法,抽象类不能进行实例化,那么要构造方法有什么作用呢?
用于子类访问父类数据时的初始化
d:抽象类不能直接实例化那么,抽象类如何实例化呢?
按照多态的方式,由具体的子类实例化。其实这也是多态的一种,抽象类多态。
e:抽象类的子类
要么是抽象类
要么重写抽象类中的所有抽象方法
C:案例演示
抽象类特点
public class Test {
public static void main(String[] args) {
Animal an=new Cat();
an.eat();
an.sleep();
an=new Dog();
an.eat();
an.sleep();
}
}
abstract class Animal{
public abstract void eat();
public abstract void sleep();
}
class Cat extends Animal{
@Override
public void eat() {
System.out.println("吃鱼");
}
@Override
public void sleep() {
System.out.println("白天睡觉");
}
}
class Dog extends Animal{
@Override
public void eat() {
System.out.println("吃骨头");
}
@Override
public void sleep(){
System.out.println("晚上睡觉");
}
}
2, 抽象类的成员特点
A:抽象类的成员特点
a:成员变量:既可以是变量,也可以是常量。
b:构造方法:有。
用于子类访问父类数据的初始化。
c:成员方法:既可以是抽象的,也可以是非抽象的。
B:案例演示
抽象类的成员特点
C:抽象类的成员方法特性:
a:抽象方法 强制要求子类做的事情。
b:非抽象方法 子类继承的事情,提高代码复用性。