继承是java领域中很重要的程序体,对于java面向对象环节,如何掌握更深度的应用,以及重复的继承知识,不断提升个人在开发领域中的技术水平,因此java开发掌握很多技巧,降低开发难度,拉勾IT课小编为大家分析。
一、什么是继承
继承就是子类继承父类的特征和行为,使得子类对象(实例)具有父类的实例域和方法,或子类从父类继承方法,使得子类具有父类相同的行为。可以联系生活进行理解,相当于父亲和儿子的关系。父亲有的属性和能力在儿子身上都会有所体现。在java中,子类继承了父类,就能使用父类的静态属性和动态方法。从而对事物进行抽象。
二、为什么需要继承——【减少代码量】
1、在程序开发中,会有很多模块有相同或者相近的属性和功能,如果为每一个模块写一遍,那代码量将会成倍的增加。当将一些共有的代码提炼出来,在需要的的地方进行使用,这就减少了不必要的代码。当配合子类特有的属性方法,就能满足特定的功能需求。
2、使类与类之间产生了关系,有了这个关系,才有多态的特性。
三、继承的实现
1. 基本语法【使用extends关键字】:
2. 修饰符 class ClassName extends extend_class {
3. // 类体
}
4. 继承的特性:
1、子类拥有父类非 private 的属性、方法。
2、子类可以拥有自己的方法和属性
3、java 的继承是单继承,但是可以多重继承,单继承就是一个子类只能继承一个父类,多重继承就是父类也可以继承于其他类。
4、
5. java继承的简单实例:
父类有属性money,car,tall,name四个属性。当被继承时,这些属性都可以在子类中直接访问到。
6. public class Father {
7. public int money=10_000_000;
8. private int car=2;
9. static double tall=178;
10. protected String name="ThreePure";
11.
12. public int getCar() {
13. return car;
14. }
15.
16. public void setCar(int car) {
17. this.car = car;
18. }
19.
20. public void say(){
21. System.out.println("你们要好好学习");
22. }
23. }
有Daughter 类,继承了父类Father类,并使用了父类中的属性和方法。
public class Daughter extends Father{
//子类自身的方法
public static void run(){
System.out.println("我喜欢跑步");
}
public static void main(String[] args) {
//父亲的身高178.0cm, 静态属性也可以通过类.属性名访问。
System.out.println("父亲的身高"+Father.tall+"cm");
Daughter daughter=new Daughter();
//你们要好好学习. 子类可以直接使用父类的方法
daughter.say();
//10000000 子类可以直接使用父类的有权限的属性
System.out.println(daughter.money);
//访问父类的私有属性,要用get,set方法
System.out.println("父亲有"+daughter.getCar()+"辆车");
System.out.println("父亲的名字"+daughter.name);
//调用子类自己的方法
Daughter.run();
}
}
结果:
父亲的身高178.0cm
你们要好好学习
10000000
父亲有2辆车
父亲的名字ThreePure
我喜欢跑步
由上可知,继承只是继承了父类的非private属性方法,当要使用私有属性变量时,可以通过get/set方法进行访问。
24.
四、继承中与构造器相关
1、子类不继承父类的构造器(构造方法或者构造函数),它只是调用。
2、如果父类构造器没有参数,则在子类的构造器中不需要使用 super 关键字调用父类构造器,系统会自动调用父类的无参构造器。
案例:父类,构造方法没有参数:
public class Father {
public int money=10_000_000;
private int car=2;
public static double tall=178;
protected String name="ThreePure";
public Father() {
System.out.println("父类的构造方法被调用了");
}
public Father(int a) {
System.out.println("父类的有参构造方法被调用了,传入参数为"+a);
}
}
子类,构造方法没有参数:
public class Son extends Father{
public Son() {
//此处默认super()
System.out.println("子类的构造方法被调用了");
}
public Son(int a) {
System.out.println("子类的有参构造方法被调用了,传入参数为"+a);
}
}
在子类中无参的构造方法默认调用了super()方法,不需要显性调用。如果一定要写,则必须写在子类无参构造器方法体的第一行。
main方法,用于启动程序:
public class Application {
public static void main(String[] args) {
System.out.println("-----------创建对象-----------");
Son erzi1=new Son();
/*System.out.println("-----------创建对象-----------");
Son erzi2=new Son(100);*/
}
}
结果:
-----------创建对象-----------
父类的构造方法被调用了
子类的构造方法被调用了
3、如果子类构造器有参数,在创建一个子类对象时传入参数,那么系统也会自动调用父类的无参构造器。
main方法更改如下
public class Application {
public static void main(String[] args) {
/*System.out.println("-----------创建对象-----------");
Son erzi1=new Son();*/
System.out.println("-----------创建对象-----------");
Son erzi2=new Son(100);
}
}
结果:
-----------创建对象-----------
父类的构造方法被调用了
子类的有参构造方法被调用了,传入参数为100
得出结论:创建子类对象时,系统会默认调用父类的无参构造器,需要显式地使用super()。
4、如果父类的构造器带有参数,则必须在子类的构造器中显式地通过 super 关键字调用父类的构造器并配以适当的参数列表。
对子类的更改如下:
public class Son extends Father{
public Son() {
super(10);
System.out.println("子类的构造方法被调用了");
}
public Son(int a) {
super();
System.out.println("子类的有参构造方法被调用了,传入参数为"+a);
}
}
main方法更改如下:
public class Application {
public static void main(String[] args) {
System.out.println("-----------创建对象-----------");
Son erzi1=new Son();
System.out.println("-----------创建对象-----------");
Son erzi2=new Son(100);
}
}
结果:
-----------创建对象-----------
父类的有参构造方法被调用了,传入参数为10
子类的构造方法被调用了
-----------创建对象-----------
父类的构造方法被调用了
子类的有参构造方法被调用了,传入参数为100
五、super和this关键字
super关键字:我们可以通过super关键字来实现对父类成员的访问,用来引用当前对象的父类。
this关键字:指向自己的引用。
父类代码
public class Father {
protected String name="ThreePure";
public void print(){
System.out.println("Father_print");
}
}
子类
public class Son extends Father{
private String name="HJB";
public Son() {
System.out.println(super.name);
System.out.println(this.name);
test1();
}
@Override
public void print(){
System.out.println("Son_print");
}
public void test1(){
//调用上面的print()方法,输出Son
print();
//Son
this.print();
//Father。调用父类的print方法
super.print();
}
}
main方法
public class Application {
public static void main(String[] args) {
System.out.println("-----------创建对象-----------");
Son erzi1=new Son();
}
}
结果:
-----------创建对象-----------
ThreePure //父类name属性
HJB
Son_print
Son_print
Father_print //父类方法
六、object类
java中,所有类都直接或者间接继承于Object类。
在IDEA中,使用快捷键Ctrl+H,能打开Hierarchy 面板,从中我们就可以知道当前类继承于哪个类以及被哪个类继承。比如我的当前Father类,继承于Object类,被Son,Daughter两个类继承。
七、重写
在子类中创建了一个与父类中名称相同、返回值类型相同、参数列表的方法相同,只是方法体中的实现不同,以实现不同于父类的功能,这种方式被称为方法重写(override),又称为方法覆盖、方法复写。
关于重写,详情参见【Java的重写】
八、注意事项
1、final 关键字声明类可以把类定义为不能继承的,即最终类;
public final class Father {}
public class Son extends Father{}
//报错:Cannot inherit from final 'characteristic.Father'
2、 implements 关键字是表示继承接口,接口是允许多继承的。
public interface A {
public void run();
}
public interface B {
public void show();
}
public class C implements A,B {
}
3、强烈建议在重写方法前加上 @Override 注解
父类:
public class Override1 {
private void method() {
System.out.println("Override1.method()");
}
}
//子类
public class Override2 extends Override1{
private void method() {
System.out.println("Override2.method()");
}
public void myMethod(){
method();
}
public static void main(String[] args) {
Override2 a = new Override2();
a.myMethod();
}
}
结果:编译通过
Override2.method()
这里父类的method()被设置成私有属性,private 的方法是无法被子类重写的,但是在子类中尝试覆盖method()方法,并且能编译通过和输出结果。
这是因为子类的method()并不是重写方法,而仅仅是与父类method()方法重名的方法。追其原因是因为缺少了@Override注解,java编译器以为这是一个普通的方法。在Override2 类中使用快捷键Alt+Insert快捷键插入重写方法,发现private 方法无法生成重写方法。