java面向对象思想之继承
一、什么是继承
继承就是子类继承父类的特征和行为,使得子类对象(实例)具有父类的实例域和方法,或子类从父类继承方法,使得子类具有父类相同的行为。可以联系生活进行理解,相当于父亲和儿子的关系。父亲有的属性和能力在儿子身上都会有所体现。在java中,子类继承了父类,就能使用父类的静态属性和动态方法。从而对事物进行抽象。
二、为什么需要继承——【减少代码量】
1、在程序开发中,会有很多模块有相同或者相近的属性和功能,如果为每一个模块写一遍,那代码量将会成倍的增加。当将一些共有的代码提炼出来,在需要的的地方进行使用,这就减少了不必要的代码。当配合子类特有的属性方法,就能满足特定的功能需求。
2、使类与类之间产生了关系,有了这个关系,才有多态的特性。
三、继承的实现
- 基本语法【使用extends关键字】:
修饰符 class ClassName extends extend_class { // 类体 }
- 继承的特性:
1、子类拥有父类非 private 的属性、方法。
2、子类可以拥有自己的方法和属性
3、java 的继承是单继承,但是可以多重继承,单继承就是一个子类只能继承一个父类,多重继承就是父类也可以继承于其他类。
4、 - java继承的简单实例:
父类有属性money,car,tall,name四个属性。当被继承时,这些属性都可以在子类中直接访问到。
public class Father { public int money=10_000_000; private int car=2; static double tall=178; protected String name="ThreePure"; public int getCar() { return car; } public void setCar(int car) { this.car = car; } public void say(){ System.out.println("你们要好好学习"); } }
有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方法进行访问。
四、继承中与构造器相关
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
方法无法生成重写方法。