面向对象之继承以及抽象(Java实现)

回顾封装

关于面向对象三大特性,我们可以很自信的回答:封装、继承、多态

之前学习的封装,可以很直观的理解为了保护数据,我们在idea中可以用alt+insert进行一个选择

面向对象之继承以及抽象(Java实现)

constructer构造方法

getter和setter方法就是对封装的一个体现,我们一般将类中的数据设为private,这样new对象时可以防止用户对数据进行修改,而getter和setter方法的建立就是为了进行修改和获得数据

这样更能体现封装性

继承

多个类中存在相同属性和行为时,将这些内容抽取到单独一个类中,那么多个类无需再定义这些属性和行为,只要继承那一个类即可。

其中,多个类可以称为子类,单独那一个类称为父类、超类(superclass)或者基类。

继承描述的是事物之间的所属关系,这种关系是: is-a 的关系。

上面的是一个基本的概述,很明显的可以看出只要我们有一个相似的属性,我们可以提取公共性质,形成一个父类,后面再有其他的类需要进行继承这些属性和方法的时候,我们就可以不需要重写那些已经写过的方法。

我们也可以对继承的方法进行覆盖重写。具体容后再说

继承的定义:

就是子类继承父类的属性和行为,使得子类对象具有与父类相同的属性、相同的行为。子类可以直接访问父类中的非私有的属性和行为。

继承的好处:

  1. 提高代码的复用性。
  2. 类与类之间产生了关系,是多态的前提。

继承的格式:

class 父类 {
...
}
class 子类 extends 父类 {
...
}

我们在子类中进行一个变量的定义,如果定义父类中未曾出现的变量,这是可以吗?

答案是可以的,子类中出现与父类不同的变量,是可以的。

子父类中出现了同名的成员变量时,在子类中需要访问父类中非私有成员变量时,需要使用super 关键字,修饰父类成员变量,类似于之前学过的 this 。

使用格式:

super.父类成员变量名

Fu 类中的成员变量是非私有的,子类中可以直接访问。若Fu 类中的成员变量私有了,子类是不能直接访问的。通常编码时,我们遵循封装的原则,使用private修饰成员变量,那么如何访问父类的私有成员

变量呢?对!可以在父类中提供公共的getXxx方法和setXxx方法。

我们需要进行父类方法的重写,我们应该如何去操作呢?

如果子类父类中出现重名的成员方法,这时的访问是一种特殊情况,叫做方法重写 (Override)。

方法重写 :子类中出现与父类一模一样的方法时(返回值类型,方法名和参数列表都相同),会出现覆盖效果,也称为重写或者复写。声明不变,重新实现。

建议:在子类重写的方法体上@override

注:

  1. 子类方法覆盖父类方法,必须要保证权限大于等于父类权限。
  2. 子类方法覆盖父类方法,返回值类型、函数名和参数列表都要一模一样。

重载(overroad):可以不同的参数列表

关于构造方法:

  1. 构造方法的名字是与类名一致的。所以子类是无法继承父类构造方法的。
  2. 构造方法的作用是初始化成员变量的。所以子类的初始化过程中,必须先执行父类的初始化动作。子类的构造方法中默认有一个super() ,表示调用父类的构造方法,父类成员变量初始化后,才可以给子类使用(super方法只能出现在构造方法的第一行)

super和this的含义

super :代表父类的存储空间标识(可以理解为父亲的引用)。

this :代表当前对象的引用(谁调用就代表谁)。

super和this的用法

  1. 访问成员
this.成员变量 ‐‐ 本类的
super.成员变量 ‐‐ 父类的
this.成员方法名() ‐‐ 本类的
super.成员方法名() ‐‐ 父类的
  1. 访问构造方法
this(...) ‐‐ 本类的构造方法
super(...) ‐‐ 父类的构造方法

子类的每个构造方法中均有默认的super(),调用父类的空参构造。手动调用父类构造会覆盖默认的super()。super() 和 this() 都必须是在构造方法的第一行,所以不能同时出现。

继承的特点

Java是单继承,不是多继承,也就是说一个类只允许有一个父亲

java支持多层继承

class A{}
class B extends A{}
class C extends B{}

最顶层的父类(祖宗)也就是object类

抽象类

父类中的方法,被它的子类们重写,子类各自的实现都不尽相同。那么父类的方法声明和方法主体,只有声明还有意义,而方法主体则没有存在的意义了。我们把没有方法主体的方法称为抽象方法。Java语法规定,包含抽象方法的类就是抽象类。

定义:

  • 抽象方法 : 没有方法体的方法。
  • 抽象类:包含抽象方法的类。

抽象的单词是abstract,所以抽象的方法也用abstract进行一个修饰。

抽象方法只有方法名,没有方法体

定义格式:

修饰符 abstract 返回值类型 方法名 (参数列表);

代码举例:

public abstract void run();

抽象类

如果一个类包含抽象方法,那么这个类必须是抽象类

定义格式:

abstract class 类名字 {
}

代码举例:

public abstract class Animal {
public abstract void run();
}

继承抽象类的子类必须重写父类所有的抽象方法。否则,该子类也必须声明为抽象类。最终,必须有子类实现该父类的抽象方法,否则,从最初的父类到最终的子类都不能创建对象,失去意义。

注意事项:

关于抽象类的使用,以下为语法上要注意的细节,虽然条目较多,但若理解了抽象的本质,无需死记硬背。

  1. 抽象类不能创建对象,如果创建,编译无法通过而报错。只能创建其非抽象子类的对象。
  2. 抽象类中,可以有构造方法,是供子类创建对象时,初始化父类成员使用的。
  3. 抽象类中,不一定包含抽象方法,但是有抽象方法的类必定是抽象类。
  4. 抽象类中,不一定包含抽象方法,但是有抽象方法的类必定是抽象类。
  5. 抽象类的子类,必须重写抽象父类中所有的抽象方法,否则,编译无法通过而报错。除非该子类也是抽象类。

关于继承的小案例,发红包

这里给出具体代码,加深理解,我就不进行诠释了

Person类:

package cn.qioha.test;

public class Person {
    private String name;
    private  int money;

    public Person() {
    }

    public Person(String name, int money) {
        this.name = name;
        this.money = money;
    }
    public void show(){
        System.out.println("my name is"+name+",money is"+money);
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getMoney() {
        return money;
    }

    public void setMoney(int money) {
        this.money = money;
    }
}

Manager类:

package cn.qioha.test;

import java.util.ArrayList;

public class Manager extends Person{
    public Manager(String name, int money) {
        super(name, money);
    }
    public ArrayList<Integer> send(int totalMoney,int count){
        ArrayList<Integer> list = new ArrayList<>();
        int money = super.getMoney();
        super.setMoney(super.getMoney()-totalMoney);
        if(totalMoney > money){
            System.out.println("your money not enough");
            return list;
        }
        int m = totalMoney / count;
        for (int i = 0; i < count - 1; i++) {
            list.add(m);
        }
        int n = totalMoney % count;
        list.add(m+n);
        return list;
    }
}

Member类:

package cn.qioha.test;

import java.util.ArrayList;
import java.util.Random;

public class Member extends Person{
    public Member(String name, int money) {
        super(name, money);
    }
    public void resive(ArrayList<Integer> list){
        int num = new Random().nextInt(list.size());
        int money = list.remove(num);
        super.setMoney(super.getMoney()+money);
    }
}

demo(实例类):

package cn.qioha.test;

import java.util.ArrayList;

public class demo {
    public static void main(String[] args) {
        Manager ma = new Manager("peter",100);
        Member q = new Member("anna",0);
        Member w = new Member("nana",0);
        Member e = new Member("qing",0);

        ma.show();
        q.show();
        w.show();
        e.show();
        ArrayList<Integer> list = new ArrayList<>();
        list = ma.send(40,3);
        q.resive(list);
        w.resive(list);
        e.resive(list);

        System.out.println("================");
        ma.show();
        q.show();
        w.show();
        e.show();
    }
}

doc文档注释我就没有进行一个撰写了,因为比较的简单,希望各位同行多多指教!

原文 

http://www.cnblogs.com/hgao/p/13251876.html

本站部分文章源于互联网,本着传播知识、有益学习和研究的目的进行的转载,为网友免费提供。如有著作权人或出版方提出异议,本站将立即删除。如果您对文章转载有任何疑问请告之我们,以便我们及时纠正。

PS:推荐一个微信公众号: askHarries 或者qq群:474807195,里面会分享一些资深架构师录制的视频录像:有Spring,MyBatis,Netty源码分析,高并发、高性能、分布式、微服务架构的原理,JVM性能优化这些成为架构师必备的知识体系。还能领取免费的学习资源,目前受益良多

转载请注明原文出处:Harries Blog™ » 面向对象之继承以及抽象(Java实现)

赞 (0)
分享到:更多 ()

评论 0

  • 昵称 (必填)
  • 邮箱 (必填)
  • 网址