面向对象中的
继承:
好处:提高代码的复用性
让类和类之间产生关系,有了这个关系,才有了多态性。
弊端:他打破了封装性
注意啦:
不要为了获取其他类的功能,简化代码而继承。必须是类与类之间有所属关系才可以继承,所属关系is a 。 (学生继承人,不能继承工人。)
Java语言当中,只支持单继承,不支持多继承。多继承容易带来安全隐患(当多个父类中定义了相同的功能,但功能内容不同时,子类对象不确定要运行哪一个),但Java会用另一种体现形式来完成表示,多实现。
单继承:只继承一个爸爸。
Java支持多层继承(爸爸,儿子,孙子)也就是一个继承体系。
除了继承,类与类之间还有 聚集关系(has a ,XX包含XXX)。
聚合:球队中有球员,班级和学生
组合:手是人身体一部分,心脏是一部分。(球队可以少球员,但是人不能少心脏 ,这是差别)
@@@@子父类中变量的特点:当子类和父类中变量重名时,用super.xx调用父类,如果不加,则调用的是子类中的变量(其实是省略了this.XXX)。super代表的是父类对象的引用,this代表的是本类对象的引用。
package biji;
public class Fu {
int num=5;
}
package biji;
public class Zi extends Fu {
int num=6;
void show(){
System.out.println(super.num);//super 指明父类num
}
}
package biji;
public class ExtendsDemo {
public static void main(String[] args) {
Zi z=new Zi();//new对象
System.out.println(z.num+z.num);//结果12
z.show();//结果5
}
}
父类私有private 子类不能继承
@@@@子父类中函数(方法)特点 覆盖(重写)
当子类出现和父类一模一样的函数时,当子类对象调用该函数,会运行子类函数的内容。如同父类的函数被覆盖一样。这种情况是函数的一种特性:重写(覆盖)。父类的方法其实还在栈内存当中,只是没有运行
package biji;
public class Fu {
void show(){
System.out.println("Fu show");
}
}
package biji;
public class Zi extends Fu {
void show(){
System.out.println("Zi show");
super.show(); //super调用了父类的方法。
}
}
}
package biji;
public class ExtendsDemo {
public static void main(String[] args) {
Zi z=new Zi();
z.show();//结果 Zi show ,覆盖了父类中的方法
Fu show, super调用了父类的方法。
}
}
在日后的开发中,可以利用这种特性来进行功能的拓展。(手机在原有的功能上进行拓展...)。
覆盖的注意:
1.子类覆盖父类,必须保证子类的权限大于等于父类权限,才可以覆盖,否则编译失败。目前有三种权限 public private 和 默认权限(其实就是什么也不加)。默认权限 介于Public和private之间。
2.静态只能覆盖静态。不能覆盖非静态。
另外,重载:只看同名函数的参数列表,重写:子父类方法要一模一样。
@@@@子父类中的构造函数
在对子类对象进行初始化时,父类的构造函数也会运行,那是因为子类的构造函数默认第一行有一条隐藏的语句 super();这条语句会访问父类中空参数的构造函数,是空参数哦!而且子类中所有的构造函数默认第一行都是super();
子类一定要访问父类中的构造函数
package biji;
public class Fu {
Fu(){
System.out.println("Fu show");
}
}
package biji;
public class Zi extends Fu {
Zi(){
//super(); 隐式的语句
System.out.println("Zi show");
}
}
package biji;
public class ExtendsDemo {
public static void main(String[] args) {
Zi z=new Zi();
//运行结果Fu show (他先出现,先运行)
Zi show (他后出现)
}
}
当父类中没有空参数的构造函数,那么就把隐藏的super();显示出来
package biji;
public class Fu {
Fu(int x){
System.out.println(x);
}
}
package biji;
public class Zi extends Fu {
Zi(){
super(9);//显示出来
System.out.println("Fu show");
}
Zi(int x){
super(1);//显示出来
System.out.println(x);
}
}
package biji;
public class ExtendsDemo {
public static void main(String[] args) {
Zi z=new Zi();
Zi z1=new Zi(4);
//结果为9 Fu show 1 4
}
}
为什么子类一定要访问父类中的构造函数?
因为父类中的数据子类可以直接获取,所以子类对象在建立的时候,需要先先查看父类是如何对这些数据进行初始化的。所以子类在对象初始化时,要先访问父类中的构造函数。
如果要访问父类中指定的构造函数,可以通过手动定义super语句的方式来指定。
package biji;
public class Person {
String name;
Person (String name){
this.name=name;
}
void show(){
}
}
package biji;
public class Student extends Person {
Student(String name){
super(name);//访问父类中的指定构造函数、调用一般函数用super.XX
} 构造函数用super(XXX)。super放在构造函数第一行。
void method(){
super.show();
}
}
结论:
子类的所有的构造函数,默认都会访问父类中空参数的构造函数。以为子类每一个构造函数内的第一行都会有一句隐式super();
当父类中没有空参数的构造函数的时候,子类必须通过手动super语句形式来指定要访问父类中的构造函数。
当然,子类的构造函数第一行也可以手动指定this语句来访问本类中的构造函数。子类中至少会有一个构造函数会访问父类中的构造函数。
父类中也有super();父类的父类是Object,所有类的父类。还有,super 和this是不能同时存在在一起的,因为他们都要在第一行,为什么在第一行?因为初始化动作要先做。
---------------------- <a href="http://edu.csdn.net"target="blank">ASP.Net+Android+IOS开发</a>、<a href="http://edu.csdn.net"target="blank">.Net培训</a>、期待与您交流! ----------------------