面向对象(二)

---------------------- 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 期待与您交流! ----------------------

 
 
 
 
 
原文地址:https://www.cnblogs.com/Lt-Java13/p/3266711.html