Java面向对象编程之类的继承详解
本文实例讲述了java面向对象编程之类的继承。分享给大家供大家参考,具体如下:
继承:特殊类拥有一般类的全部属性与行为。
继承好处:
1.提高了代码的复用性
2.让类与类之前产生了关系,有了这个关系才有多态的特性。继承是类和类之前的关系。
注意事项:
1.java只支持单继承,不支持多继承。因为多继承有安全隐患:当多个父类定义相同的函数,但是功能不同时,子类不知道运行哪一个。
2.子类继承父类时,继承了父类的所有方法和属性,可直接使用。
3,java支持多层继承,即:孙-子-父的关系
语法:
[类修饰符] 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;}//不可以,静态只能覆盖静态。<br><br>因此子类允许重写和重载。
更多java相关内容感兴趣的读者可查看本站专题:《java面向对象程序设计入门与进阶教程》、《java数据结构与算法教程》、《java操作dom节点技巧总结》、《java文件与目录操作技巧汇总》和《java缓存操作技巧汇总》
希望本文所述对大家java程序设计有所帮助。
上一篇: 高吞吐、线程安全的LRU缓存详解