目录
extends
Java关键字 用作继承
1.类中的声明,通过extends来创建父类的子类(Java继承具有单继承的特点,每个子类只有一个直接父类.声明为final的不可被继承)
Java里子类继承父类的语法格式如下:
- 修饰符 class AClass extends ASuperClass
- {
- //类定义的部分
- }
子类继承父类, 将获得父类的全部成员变量和方法.
现实中子辈从父辈那里获得一笔财富的继承关系很像.
但是, Java的子类不能继承父类的构造器.
下面写个程序示范子类继承父类.
- public class Food
- {
- public int weight;
- public void say()
- {
- System.out.println("这个食物重:" + weight + "g!");
- }
- }
接下来定义该Food类的子类Noodle
- public class Noodle extends Food
- {
- public static void main(String[] args)
- {
- //创建Noodle对象
- Noodle n = new Noodle();
- //Noodle 对象本身没有 weight 成员变量
- //因为Noodleo 继承Food父类有 weight 成员变量, 所以访问 Noodle 对象也有 weight 成员变量.
- n.weight = 50;
- //调用 Noodle 对象的 say() 方法
- n.say();
- }
- }
implements
Java关键字 用作实现
1.类通过implements声明自己使用的接口,接口的方法一般为空的, 必须重写才能使用(可以是一个或者多个接口)
Java里类实现接口的语法格式如下:
例子来源于网络
- public inerface Runner
- {
- int ID = 1;
- void run ();
- }
注:接口中的方法都是抽象的。
- interface Flyer
- {
- void fly ();
- }
- class Bird implements Runner , Flyer
- {
- public void run ()
- {
- System.out.println("the bird is running");
- }
- public void fly ()
- {
- System.out.println("the bird is flying");
- }
- }
注意:1.多个类可以实现同一个接口,一个类可以实现多个接口。
2.实现一个接口,要实现该接口的所有的方法(抽象类除外);
3.在看API文档时,发现一个接口可以是其他接口的子接口,说明接口之间会存在继承的关系
总结:
extends与implements的不同
extends是继承父类,继承只能继承一个类。JAVA中不支持多重继承。
implements可以用接口来实现,用到implements来实现多个接口,用逗号分开就行了
例:
class A extends B implements C,D,E(class 子类名 extends 父类名 implenments 接口名)
浅谈一下使用extends和implements后父类、子类发生的变化
extends 可以实现父类,也可以调用父类初始化 this.parent()。而且会覆盖父类定义的变量或者函数。这样的好处是:架构师定义好接口,让工程师实现就可以了。整个项目开发效率和开发成本大大降低。
implements,实现父类,子类不可以覆盖父类的方法或者变量。即使子类定义与父类相同的变量或者函数,也会被父类取代掉。
这两种实现的具体使用,是要看项目的实际情况,需要实现,不可以修改implements,只定义接口需要具体实现,或者可以被修改扩展性好,用extends。
说说接口:
接口一般是只有方法声明没有定义的, 接口是没有实现自己的方法,仅仅有声明也就是一个方法头没有方法体 接口是子类实现其方法声明而不是继承其方法;
拓展一下学习extends 和 implements的其他博文见解
父类A与子类B继承关系上的不同:
A a = new B(); 结果a是一个A类的实例,只能访问A中的方法,那么又和A a = new A();有什么区别呢?
class B extends A
继承过后通常会定义一些父类没有的成员或者方法。
A a = new B();
这样是可以的,上传。
a是一个父类对象的实例,因而不能访问子类B定义的新成员或方法。
假如这样定义:
class A
{
int i;
void f(){}
}
class B extends A
{
int j;
void f(){} //重写
void g(){}
}
然后:
B b = new B();
b就是子类B对象的实例,不仅能够访问自己的属性和方法,也能够访问父类的属性和方法。诸如b.i,b.j,b.f(),b.g()都是合法的。此时b.f()是访问的B中的f()
A a = new B();
a虽然是用的B的构造函数,但经过upcast,成为父类对象的实例,不能访问子类的属性和方法。a.i,a.f()是合法的,而a.j,a.g()非法。此时访问a.f()是访问B中的f()
A a = new B(); 这条语句,实际上有三个过程:
(1) A a;
将a声明为父类对象,只是一个引用,未分配空间
(2) B temp = new B();
通过B类的构造函数建立了一个B类对象的实例,也就是初始化
(3) a = (A)temp;
将子类对象temp转换为父类对象并赋给a,这就是上传(upcast),是安全的。
经过以上3个过程,a就彻底成为了一个A类的实例。
子类往往比父类有更多的属性和方法,上传只是舍弃,是安全的;而下传(downcast)有时会增加,通常是不安全的。
此时作者的心里有一万只草泥马在奔腾,大概我的理解就是,给谁分配了空间就是访问谁的方法;
a.f()对应的应该是B类的方法f()
调用构造函数建立实例过后,对应方法的入口已经确定了。
如此以来,a虽被上传为A类,但其中重写的方法f()仍然是B的方法f()。也就是说,每个对象知道自己应该调用哪个方法。
A a1 = new B();
A a2 = new C();
a1,a2两个虽然都是A类对象,但各自的f()不同。这正是多态性的体现。
其实不管是extends 还是implements都是多态性的一种体现
父类A与子类B继承关系上的不同见博文https://www.cnblogs.com/dengyungao/p/7524955.html