JAVA基础之继承

继承的意思就是,当很多类的成员变量相同时,可以将相同的成员变量单独抽取出来
编辑一个新的类。其余的类需要这些成员变量时,直接继承即可。

继承关键词: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.初始化完毕后,将地址值赋给引用变量。





评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值