继承:特殊类拥有一般类的全部属性与行为。
继承好处:
1.提高了代码的复用性
2.让类与类之前产生了关系,有了这个关系才有多态的特性。继承是类和类之前的关系。
注意事项:
1.java只支持单继承,不支持多继承。因为多继承有安全隐患:当多个父类定义相同的函数,但是功能不同时,子类不知道运行哪一个。
2.子类继承父类时,继承了父类的所有方法和属性,可直接使用。
3,java支持多层继承,即:孙-子-父的关系
4.千万不要为了获取其他类的功能,简化代码而继承,继承必须是类和类之前所属关系,子类有父类的所有方法和属性,才选择继承。
语法:
[类修饰符] class 子类名 extends 父类名{ 语句; }
例如:
class Pserson { int age; String name; public void speak() { System.out.println("Hello World!"); } } //继承Person类,继承了父类所有方法和属性 class Student extends Pserson { public void study() { System.out.println("Good Study!"); } } //继承Person类,继承了父类所有方法和属性 class Worker extends Pserson { public void work() { System.out.println("Good work!"); } }
如何使用一个继承体系中的功能(查阅api文档):
查阅父类的功能,创建子类对象使用功能
在继承过程中经常遇到这三种场景:
1)同名变量
1.如果子类出现非私有的同名成员变量时,子类访问本类的变量,用this;子类访问父类中的同名变量,用super。
2.this代表本类对象的引用
3.super代表父类对象的引用(用法和this相同)
2)同名函数
1.如果子类出现和父类一模一样的函数时(函数名和参数都相同),当子类对象调用该函数,会运行子类函数内容。,父类的函数会被覆盖(也叫重写)。
2.重写定义:当子类继承父类,沿袭了父类的功能,到子类中。但子类虽具备该功能,但功能的内容和父类不一致,这时,没有必须要定义新功能,而是使用覆盖特性,保留父类的功能定义,并重写功能内容。
3.重写(覆盖)注意事项:
<1>子类覆盖父类,必须保证子类的权限大于等于父类的权限,才能继承,否则编译失败。(public>不写修辞关键词>private)
<2>静态只能覆盖静态
<3>重载:只看同名函数的参数列表 重写:子父类方法要一模一样(函数名和参数列表,返回值类型)
class Fu { //public void show() 当父类为show()时,会和子类函数一模一样,父类的show函数会被重写 public void show(String name) //父类的show函数和子类不一样(参数列表不一样),因此父类的show函数不会被重写 { System.out.println(name); } } class Zi extends Fu { public void show() { System.out.println("zi"); } } class Jicheng { public static void main(String[] args) { Zi z1=new Zi(); z1.show("nihao");//会调用父类的show函数 } }
3)构造函数
1.在对子类对象进行初始化时,父类的构造函数也会运行,因为子类的构造函数的第一行默认有一条隐式语句super()
2.super()会访问父类中空参数的构造函数,而且子类中所有的构造函数第一行默认都是super()
3.子类一定要访问父类构造函数原因
<1>因为父类中的数据子类可以直接获取,所以子类在创建是,先看看父类如何对这些数据进行初始化的,所以子类在对象初始化时,默认先访问父类的构造函数。
<2>若要访问父类制定的构造函数或者父类没有空参数的构造函数时,可以通过手动定义super语句的方式来制定。
<3>当然子类的构造函数第一行也可以手动指定this语句来访问本类的构造函数,但子类中至少有一个构造函数会访问父类的构造函数
class Fu { String name; int age; Fu(){System.out.println("Hello Fu");} Fu(String name) { System.out.println(name); } Fu(String name,int age) { this.name=name; this.age=age; System.out.println("name: "+name+",age: "+age); } } class Zi extends Fu { //Zi(){System.out.println("Hello Zi");} 默认先会调用父类的无参构造函数 Zi() { super("zhangsan",20);//手动用super语句指定父类的构造函数,来获取父类非私有的信息 System.out.println(name+"::"+age); } } class Test { public static void main(String[] args) { Zi z1=new Zi(); } }
构造函数异常例子:
写出程序结果 class Super { int i=0; public Super(String s) { i=1; } } class Demo extends Super { public Demo(String s) { i=2; } public static void main(String[] args) { Demo d=new Demo("yes"); System.out.println(d.i); } } //编译失败,因为父类中缺少空参数的构造函数。 //或者子类应该通过super语句指定要调用的父类中的构造函数。
重写和重载例子:
class Demo { int show(int a,int b){return 0;} } 下面那些函数可以存在于Demo的子类中。 A.public int show(int a,int b){return 0;}//可以,覆盖。 B.private int show(int a,int b){return 0;}//不可以,权限不够。 C.private int show(int a,long b){return 0;}//可以,和父类不是一个函数。没有覆盖,相当于重载。 D.public short show(int a,int b){return 0;}//不可以,因为该函数不可以和给定函数出现在同一类中,或者子父类中。 E.static int show(int a,int b){return 0;}//不可以,静态只能覆盖静态。
因此子类允许重写和重载。