您的位置:首页 > 编程语言 > Java开发

JAVA基础之继承

2016-05-22 14:23 357 查看
继承的意思就是,当很多类的成员变量相同时,可以将相同的成员变量单独抽取出来

编辑一个新的类。其余的类需要这些成员变量时,直接继承即可。

继承关键词:extends

比如:

class FuLei	//这是父类。将下面相同的成员变量name和age抽取出来。
{
String name;
int age;
}
class ZiLei extends/*继承*/ FuLei//这是子类,添加上继承关键词和继承类名,就可应用父类的成员变量。
{
void zilei()
{
System.out.println(name+":hello"+age);
}

}
class ZiLei2 extends FuLei//同理。
{
void zilei()
{
System.out.println(name+":hell,Google"+age);
}
}
继承的好处:

1.提高了代码的复用性。

2.让类与类之间产生关系,给第三个特征多态提供了前提。

Java中支持单继承,不支持多继承,但对C++中的多继承进行改良。

单继承:一个子类只能有一个直接父类。

多继承:一个子类可以有多个直接父类。(Java中不允许,但有改良方法。)

比如:

class a
{
void show()
{
System.out.print();
}
}
class b
{
void show()
{
System.out.print();
}
}
class c extends a,b
{
}
以上代码就是C++的多继承,但是有个弊端,

c类继承了a和b类,假如要调用show ()方法时,就无法分辨具体调用的是哪个父类的show ()方法。

所以 Java不直接支持多继承,因为多个父类有相同成员时,会产生调用的不确定性。

在Java中是通过“多实现”的方式来体现。(后面会讲到)

Java支持多层(多重)继承。

即C继承B,B继承A。

这样的继承方式,也叫继承体系。

当要使用一个继承体系时,

1.查看该体系中的顶层类,了解该体系的基本功能。

2.创建体系的最子类对象,完成功能的使用。

什么时候定义继承:

当类与类之间存在着所属关系时,就定义继承。

在子父类中,成员的特点体现:

1.成员变量。

2.成员函数。

3.构造函数。

1.成员变量。

class Fu
{
int num=5;
}
class Zi extends Fu
{
int num=7;
void show()
{
System.out.println(num+"-----"+super.num);
}
}
以上代码可以看出,当子类的成员变量和父类的成员变量同名时,为了进行区分,我们加上了super关键词。

我们得出结论:

当本类的成员和局部变量同名时用this区分。

当子父类中的成员变量同名时用super区分父类。(super代表父类。)

this:代表一个本类对象的引用。

super:代表一个父类空间。

2.成员函数。

当子父类中出现成员函数一模一样(不仅仅是同名。连返回值类型,参数列表完全一样)的情况,

会运行子类的函数。

这种现象,称为覆盖操作(内存中不是覆盖操作,只是一种称呼)。这是函数在子父类中的特性。

函数的两个特性:

1.重载。同一个类中。

2.覆盖。子类中,覆盖也称为重写,覆写。

覆盖注意事项:

1.子类方法覆盖父类方法时,子类权限必须要大于等于父类的权限。

2.静态函数只能覆盖静态函数,或被静态函数覆盖。

权限大小依次是:public> (默认为空,无修饰符)>private

覆盖示例:

class Fu1
{
void show()
{
System.out.println("当我遇上你");
}
}
class Zi1 extends Fu1
{

void show()
{
System.out.println("口琴别恋");//因为覆盖操作,系统会输出的是子类的函数。
}
}
什么时候使用覆盖操作:

当对一个类进行子类扩展时,子类需要保留父类的功能声明。

但是要定义子类中该功能的特有内容时,就使用覆盖操作。

3.构造函数。

创建子类对象时,不仅会运行子类的构造函数,还会运行父类的空参数构造函数。

是因为子类的构造函数中第一行有一个默认的隐式语句: super();

这个动作就叫做子类的实例化过程。

子类的实例化过程:子类中所有的构造函数默认都会访问父类中的空参数函数。

子类实例化时候访问父类的构造函数的原因:

因为子类继承了父类,获取到了父类中内容(属性)。所以在使用父类内容之前,

要先看父类是如何对自己的内容进行初始化的。

所以子类在创建对象时,必须访问父类的构造函数。

为了完成这个必须的动作,就在子类的构造函数中加入了super();语句。

如果父类中没有定义空参数构造函数,那么子类的构造函数必须用super明确要调用父类的哪个构造函数。

同时子类构造函数中如果使用了this();语句调用了本类构造函数时,

那么super语句就没有了,因为super和this都只能定义在第一行,所以只能有一个。

但是可以保证的是,子类中肯定会有其他的构造函数访问父类的构造函数。

注意:

super语句必须要定义在子类构造函数的第一行。因为父类的初始化动作要先完成。

比如:

class Fu2
{
Fu2()//这是父类的空参数构造函数。
{
System.out.println("《好久不见》-陈奕迅");
}
}
class Zi2 extends Fu2
{
Zi2()//这是子类的空参数构造函数。
{
//super(); //这是默认调用的父类空参数构造函数,是隐式语句。不写也会有。
System.out.println("《不如不见》-Eason");
}
}
super();隐式语句的特点:、

当父类中没有定义构造函数或定义的是空参数构造函数时,该语句就会顺利执行父类的构造函数。

当父类中定义的是有参数的构造函数时,语句执行就会报错。

因为定义了构造函数后,默认的构造函数就没有了。

这时候要运行父类中定义的构造函数(即有参数的构造函数),就需要手动操作。

比如:super(参数);
程序就会顺利运行。

比如:

class Fu3
{
Fu3(int a)
{
System.out.println("《好久不见》-陈奕迅");
}
}
class Zi3 extends Fu3
{
Zi3()
{
super(3);
System.out.println("《不如不见》-Eason");
}
}
一个对象的实例化过程:(创建过程)

Person p= new Person();

1.JVM会读取指定路径下的Person.class文件,并加载进内存,

并会先加载Person的父类。(如果有直接父类的话)

2.在堆内存中开辟空间,分配地址。

3.并在对象空间中,对对象中的属性进行默认初始化。

4.调用对应的构造函数进行初始化。

5.在构造函数中,第一行会先调用父类中构造函数进行初始化。

6.父类初始化完毕后,再对子类的属性进行显示初始化。

7.在进行子类的构造函数的特定初始化。

8.初始化完毕后,将地址值赋给引用变量。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: