面向对象(二)
---------------------- QQ:371524846 期待与您交流! ----------------------
一、继承
继承是面向对象三大特征之一,也是实现软件复用的重要手段。
1、继承概述:
子类会自动继承父类所有方法和属性,称为继承。
父类是从子类不断往上抽取出来的。
2、继承的作用:
提高了代码的复用性。
让类与类之间产生了关系。有了这个关系,才有了多态的特性。
注意:千万不要为了获取其他类的功能,简化代码而继承。
必须是类与类之间有所属关系才可以继承。所属关系is a。
Java语言中:java只支持单继承,不支持多继承。
3、继承的缺点:
因为多继承容易带来安全隐患:当多个父类中定义了相同功能,
当功能内容不同时,子类对象不确定要运行哪一个。也就是说多继承的缺点即是引发二义性。
但是Java保留这种机制。并用另一种体现形式来完成--------多实现。
Java支持多层继承,也就是一个继承体系。
如何使用一个继承体系中的功能呢?
想要使用体系,先查阅体系中父类的描述,因为父类中定义的是该体系中共性功能。
通过了解共性功能,就可以知道该体系的基本功能。
那么这个体系已经可以基本使用了。
那么在具体调用时,要创建最子类的对象,为什么呢?
一是因为有可能父类不能创建对象;
二是创建子类对象可以使用更多的功能,包括基本的也包括特有的。
注意:
构造函数是继承不了的。
Private的属性和方法也是无法继承的。
简单一句话:查阅父类功能,创建子类对象使用功能。
eg:
1 class C{ 2 demo1(){} //所提取到的共性的方法 3 } 4 class A extends C{ 5 //demo1(){} 6 demo2(){} 7 } 8 class B extends C{ 9 //demo1(){} 10 demo2(){} 11 }
4、子父类出现后,类成员的特点:
4.1 类中成员:
变量。
函数。
构造函数。
构造代码块
变量:
如果子类中出现非私有的同名成员变量时,
子类要访问本类中的变量,用this。
子类要访问父类中的同名变量,用super。
super的使用和this的使用几乎一致。
this代表的是本类对象的引用。
super代表的是父类对象的引用。
eg:
1 class Fu{ 2 int num = 4; 3 } 4 class Zi extends Fu{ 5 int num = 5; 6 void show(){ 7 System.out.println(num);//this引用,此处省略this,结果是5 8 System.out.println(super.num); //super引用,结果是3. 9 } 10 } 11 public class ExtendsLearn 12 { 13 public static void main(String[] args){ 14 Zi z = new Zi(); 15 z.show(); 16 } 17 }
5、子父类中的函数
5.1覆盖(重写)
什么情况下用到覆盖呢?
当子类继承父类,沿袭了父类的功能,到子类中,
但是子类虽具备该功能,但是功能的内容却和父类不一致,
这时,没有必要定义新功能,而是使用覆盖特性,保留父类的功能定义,并重写功能内容。
覆盖(注意事项):
(1)子类覆盖父类,必须保证子类权限大于等于父类权限,才可以覆盖,否则,编译失败。
(2)静态只能覆盖静态。(若出现这种情况就和多态没关系了,编译器直接做选择了)
eg:
1 class Fu { 2 public int field = 10; 3 public void fun() { 4 System.out.println("Hello A"); 5 } 6 public static void sFun() { 7 System.out.println("hello A"); 8 } 9 } 10 class Zi extends Fu { 11 public int field = 20; //重写父类属性 12 //重写父类函数 13 public void fun() { 14 System.out.println("Hello B"); 15 } 16 public static void sFun() { //重写父类函数 17 System.out.println("hello B"); 18 } 19 } 20 public class MainApp { 21 public static void main(String[] args) { 22 Fu f1 = new Zi(); 23 f1.fun(); // Hello B 24 f1.sFun(); // Hello A 25 System.out.println(f1.field); // 10 26 } 27 }
在内存中的加载方式:
5.2 子父类中的构造函数:
在对子类对象进行初始化时,父类的构造函数也会运行,
那是因为子类的构造函数默认第一行有一条隐式的语句super();
super():会访问父类中空参数的构造函数。而且子类中所有的构造函数默认第一行都是super();
注意:
a.如果父类中无空参数构造函数(即父类中显式的构造了含有参数的构造函数),必须手动加入父类中该构造函数。
b.构造函数不存在覆盖,子类中的构造函数必定至少有一个会访问父类中的构造函数。
eg:
1 class Fu { 2 Fu(){} 3 } 4 class Zi extends Fu { 5 Zi(){ 6 super(); //不带参数,可以隐藏,不用手动添加super() 7 } 8 } 9 ------------------------------------------------------------------ 10 class Fu { 11 Fu(int x){} 12 } 13 class Zi extends Fu { 14 Zi(){ 15 super(x); //必须手动添加super(x),因为带参数。 16 } 17 }
总结:子类实例化过程
a. 子类所有的构造函数默认均访问父类中的空参数构造函数,因为子类每个构造函数内的第一行均有一句隐式的super();语句。
b. 当父类中悟空参数构造函数时,子类必须手动通过super语句的形式指定要访问的父类中的构造函数
c. 子类构造函数也可手动指定this语句,来访问本类中的构造函数,但子类中至少要有一个构造函数访问父类的构造函数。
6、final修饰符
(1)可以修饰类、函数、变量。
(2)被final修饰的类不可以被继承,为了避免被继承,被子类复写功能。
(3)被final修饰的方法不可以被复写。
(4)被final修饰的变量是一个常量只能赋值一次,既可以修饰成员变量,有可能修饰局部变量。当在描述事物时,
一些数据的出现值是固定的,那么这时为了增强阅读性,都给这些值起个名字,方便阅读。而这个值不需要改变,所以加上final修饰。
作为常量:常量的书写规范所有字母都大写,如果由多个单词组成,单词见通过_链接。
(5)内部类定义在类中的局部位置上是,只能访问该局部被final修饰的局部变量。
二、抽象类
1、抽象类的含义:
当多个类中出现相同功能,但是功能主体不同,这时可以进行向上抽取,这时只抽取功能定义,而不抽取功能主体。
2、抽象类的特点:
(1)抽象方法一定在抽象类中。
(2)抽象方法和抽象类都必须被abstract关键字修饰。
(3)抽象类不可以用new创建对象。因为调用抽象方法没意义。
(4)抽象类中的抽象方法要被使用,必须由子类复写起所有的抽象方法后,建立子类对象调用。
如果子类只覆盖了部分抽象方法,那么该子类还是一个抽象类。
eg:
1 abstract class Person 2 { 3 private String name; 4 public Person(String name) 5 { 6 this.name = name; 7 } 8 //抽象方法:对人的描述,但对每种人的描述是不清楚的,所以用抽象方法 9 public abstract String getInform(); 10 11 public String getName() 12 { 13 return name; 14 } 15 } 16 //Student继承父类Person 17 class Student extends Person 18 { 19 public Student(String name) 20 { 21 super(name); 22 } 23 //重写父类的抽象方法,对学生进行具体描述 24 public String getInform() 25 { 26 return "I'm a student."; 27 } 28 } 29 //Teacher继承父类Person 30 class Teacher extends Person 31 { 32 public Teacher(String name) 33 { 34 super(name); 35 } 36 //重写父类的抽象方法,对老师进行具体描述 37 public String getInform() 38 { 39 return "I'm a Teacher."; 40 } 41 } 42 //测试 43 public class PersonTest 44 { 45 public static void main(String[] args) 46 { 47 Person[] p = new Person[2];//用抽象类创建数组 48 //再创建两个对象存入数组 49 p[0] = new Student("Nan"); 50 p[1] = new Teacher("Wan"); 51 System.out.println(p[0].getName() + ":" + p[0].getInform()); 52 System.out.println(p[1].getName() + ":" + p[1].getInform()); 53 } 54 } 55 在这里,Person[] p = new Person[2]只是创建了两个数组类型的元素,并不是Person类对象, 56 在这里只是一种引用。而下面的new Student和new Teacher才真正创建了两个对象,分别赋值给 57 数组中的两个元素,或者说是两个变量。
3、抽象类和一般类没有太大的不同:
该如何描述事物,就如何描述事物,只不过,该事物出现了一些看不懂的东西。这些不确定的部分,也是该事物的功能,
需要明确出现,但是无法定义主体。通过抽象方法来表示。
a)抽象类比一般类多了个抽象函数。
b)抽象类不可以实例化。
特殊:抽象类中可以不定义抽象方法,这样做仅仅是不让该类建立对象。
---------------------- QQ:371524846 期待与您交流! ----------------------