继承:特殊类拥有一般类的全部属性与行为。
继承好处:
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;}//不可以,静态只能覆盖静态。
因此子类允许重写和重载。