java的继承机制

继承的基本概念

在日常生活中,我们经常用到“是一种(IS-A)”关系来组织和表达知识,从而将知识组织成一种有层次、可分类的体系结构。例如,鸭梨是一种梨,梨是一种水果;大叶榕是一种树,树是一种植物等等,数不胜数。

在面向对象程序中,用IS-A关系来描述类与类之间的关系,称之为继承(inheritance)继承就是子类继承父类的特征和行为,使得子类对象(实例)具有父类的实例域和方法,或子类从父类继承方法,使得子类具有父类相同的行为。继承是java程序设计中的一项核心技术。

有时候,我们需要创建一个新的类A,但是它具有一个已存在的类B所具有的功能,同时也有类B没有的功能。那么,我们可以认为类A与类B存在IS-A关系,“A IS-A B”。可以将A比作树,将B比作植物,那么树具有植物的特性,同时也具有树特有的特性,以区别花、草等其他植物。

创建类A时,我们可以将类B的代码复制过来,然后再在这个基础上添加代码。但这个方法让我们感到很麻烦。我们还可以用面向对象中的继承机制来构建这个新的类,我们只需要添加那些特有的特征就行了。使用继承机制,就不必每次都从头开始定义一个新的类,而是将这个新的类作为一个或者若干个现有的类的扩充或特殊化

在以上类A和类B的例子中,类A继承了类B,那么类B称为类A的父类(parent)、超类(super-class)或基类(base),而类A称为类B的子类(child)或派生类(derived-class)。一个类的祖先类(ancestor)包括了其父类及其父类的祖先类,一个类的后代类(descendant)包括了其子类及其子类的后代类。类A继承了类B也可以说是类B派生出了类A

继承的作用

继承机制主要具有双重作用:一是作为的构造机制,二是作为类型的构造机制。

  1. 作为类的构造机制,继承通过扩充、组合现有的类来构造新的类。扩充是指形成现有类的特例——派生类,组合是指抽取出若干现有类的共性形成新的抽象层次——基类。
  2. 作为类型的构造机制,如果子类继承父类,则所有要求对象为父类类型的地方也可以接受子类类型的对象。也就是说父类对象出现的地方可以用子类对象替代

作为类的构造机制,继承是面向对象程序设计语言中支持软件重用的一种重要方式,而作为类型构造机制,继承是实现动态多态性的基础。

继承的语法

java中用保留字extends表示继承关系。继承的语法格式为:

public class 子类 extends 父类{
    //子类新增的数据成员和方法
    //子类重写父类的成员和方法
}

下面举例来说明java的继承机制,以员工和经理的关系为例。

代码清单1:Employee.java

import java.util.Date;
import java.util.GregorianCalendar;
/**
 * 员工类
 * @version v1.0 2018-03-13
* @author 叶汉伟
*/
public class Employee {
    private String name;
    private double salary;
    private Date hireDay;

/**
 * 构造函数
 * @param n 员工姓名
 * @param s 员工工资
 * @param year 雇佣年份
 * @param month 雇佣月份
 * @param day 雇佣日
 */
    public Employee(String n,double s,int year,int month,int day){
        this.name=n;
        this.salary=s;
        GregorianCalendar calendar=new GregorianCalendar(year,month-1,day);
        this.hireDay=calendar.getTime();
    }

    public String getName(){
        return this.name;
    }

    public double getSalary(){
        return this.salary;
    }

    public Date getHireDay(){
        return this.hireDay;
    }

    public void raiseSalary(double byPercent){
        double raise=salary*byPercent/100;
        salary+=raise;
    }
}

代码清单2:Manager.java

/**
 * 经理类,继承了员工类Employee,经理也是员工。
 * @version v1.0 2018-03-13
 * @author 叶汉伟
 */
public class Manager extends Employee{
    private double bonus;
    /**
     * 构造函数
     * @param n 员工姓名
     * @param s 员工工资
     * @param year 雇佣年份
     * @param month 雇佣月份
     * @param day 雇佣日
     */
    public Manager(String name,double s,int year,int month,int day){
        super(name, s, year, month, day);
        this.bonus=0;
    }
    //经理的工资=基本工资+奖金
    public double getSalary(){
        double baseSalary=super.getSalary();
        return baseSalary+bonus;
    }
    //奖金
    public void setBonus(double b){
        bonus=b;
    }
}

代码清单3:ManagerTest.java

public class ManagerTest {
    public static void main(String[] args){
        Manager boss=new Manager("Carl Cracker",80000,1987,12,15);
        boss.setBonus(5000);
        
        Employee[] staff=new Employee[3];
        
        staff[0]=boss;
        staff[1]=new Employee("Harry Hacker",50000,1989,10,1);
        staff[2]=new Employee("Tommy Tester",40000,1990,3,15);
        
        for(Employee e:staff)
            System.out.println("name:"+e.getName()+",salary="+e.getSalary());
    }
}

运行结果:

name:Carl Cracker,salary=85000.0 </br>
name:Harry Hacker,salary=50000.0 </br>
name:Tommy Tester,salary=40000.0

在以上程序中,我用Employee类型的对象staff[0]引用了Manager类型的对象,程序仍能正常运行,说明了程序中任何出现超类的地方都可以用其子类置换。在此例中,staff[0]与boss引用同一个对象,但编译器仍认为staff[0]是Employee类型的对象,boss为Manager类型的对像,所以staff[0]不能用Manager类的方法。如staff[0].setbounds(5000);是错误的用法,编译器会报错。

继承成员访问控制

继承机制引入了受保护(protected)成员,提供了一种新的成员访问控制级别,其可以理解为介于公有(public)和私有(private)之间。

在继承中,子类继承了超类除构造函数之外的所有成员,这些成员成为子类的继承成员。继承成员不仅包括超类中定义的共有、受保护及私有成员,还包括了超类的继承成员。

在子类中,子类可以访问自身定义的所有成员,也可以访问父超类的共有和受保护继承成员,但不能访问超类的私有继承成员。

继承成员在子类中的访问控制与它们在超类中的访问控制相同。及原先在超类中是共有的成员,被子类继承后认为共有的成员;原先在超类中是受保护的成员,被子类继承后仍为受保护的成员;原先在超类中是私有的成员,被子类继承后认为私有的成员,但子类不能访问。数据类型为子类的对象不能访问子类的及其父类的受保护成员

重定义

当子类要用到超类中的一个方法,而超类的这个方法有不适合子类时,我们可以选择在子类中重定义这个方法。子类重定义超类的方法是指的方法是指在子类中定义一个与超类的某个方法有完全相同接口的方法,这是称子类的这个方法重定义了超类同接口的方法。所为方法接口完全相同是指返回类型方法名方法参数列表完全相同。向以上例子中,Manager类的getSalary()方法就是重定义了超类Employee的getSalary()方法。

在重定义超类(祖先类)的方法时,不允许降低方法的访问控制权限。如在超类中是public的方法,在子类重定义时不能定义成protected或private。不过java允许在子类中重定义成员数据是降低其访问控制权限。如在超类中是public的数据成员,在子类可重定义为protected或private的数据成员。

对于成员数据来说,只要子类定义了与祖先类同名的成员数据就是重定义了祖先类的成员数据,而且屏蔽了祖先类的成员数据,子类及子类的使用者再也不能访问该祖先类的成员数据。

this与super

java中的this与super都是每一个对象实例中的特殊私有成员,是一个引用变量,this的类型是该对象实例所属的类类型,super的类型是它所属的对象实例的超类类型。

它们有两种使用方式:

第一种:调用类的数据或方法成员。this是调用自身的成员,而super则是调用其超类(祖先类)中的数据成员。如以上例子的Employee类,其部分代码如下:

    public class Employee {
        private String name;
        ......
        public Employee(String name,double s,int year,int month,int day){
            this.name=name;
            .........
        }
        .......
    }

在构造函数中用到了this.name=name;,this指明了name是它所在的类(Employee)中定义的name,而不是参数列表中的那个name。这种使用方法的格式总结为:this.本类成员super.超类成员其中超类成员不能是私有成员。

第二种:调用构造函数。this是在自身的构造函数中调用自身的另一个构造函数,而super则是在自身的构造函数中调用其超类的构造函数。例如Manager类的构造函数:

    public Manager(String name,double s,int year,int month,int day){
        super(name, s, year, month, day);//必须放在第一句
        this.bonus=0;
    }

其中有super(name, s, year, month, day);,这是Manager的构造函数调用了其超类Employee的构造函数,初始化了继承成员name、salary、hireDay。这种使用方法的格式总结为:this(参数列表);super(参数列表);注意,他们都必须放在构造方法的第一句。

总结

  • 子类拥有父类非private的属性,方法。
  • 子类可以拥有自己的属性和方法,即子类可以对父类进行扩展。
  • 子类可以用自己的方式实现父类的方法(重定义)。
  • java的继承是单继承,但是可以多重继承,单继承就是一个子类只能继承一个父类,多重继承就是,例如A类继承B类,B类继承C类,所以按照关系就是C类是B类的父类,B类是A类的父类,这是java继承区别于C++继承的一个特性。但接口允许多继承
  • 提高了类之间的耦合性(继承的缺点,耦合度高就会造成代码之间的联系)。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,047评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,807评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,501评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,839评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,951评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,117评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,188评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,929评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,372评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,679评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,837评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,536评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,168评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,886评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,129评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,665评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,739评论 2 351

推荐阅读更多精彩内容