Java中的面向对象的简单介绍(四)接口和多态

因为Java中继承具有单一性,只能继承一个父类,所以为了处理这种局限性,Java又提供了接口。

接口interface

是一个比抽象类还要抽象的类,因为其内部的所有方法全部都是抽象方法。
同时接口和类的关系也不再是继承(extends)而是实现(implements)

接口的特点:

  • 只能有抽象方法
  • 只能有常量,默认使用public static final修饰
  • 方法默认就被public abstract修饰,并且也只能被public abstract修饰
  • 一样的,接口也不能实例化
  • 类和接口的关系不再是extends,而是implements,并且也要实现它所有的方法(@Override)
public class InterfaceDemo {
    public static void main(String[] args) {
        Cat c = new Cat();
        c.eat(); // 猫吃鱼
        System.out.println(Animal.num); // 3 可以直接访问接口内部的常量
    }
}

interface Animal { // 声明接口和声明类很相似
    public static final int num = 3; // 接口内部只支持使用常量,并且默认使用public static final来修饰  当然可以直接写出int num = 3;
    public abstract void eat(); // 内部的方法默认使用public abstract修饰,同时也只允许使用public abstract修饰,也可以直接写成void eat();
}

class Cat implements Animal { // 类和接口的关系是implements
    @Override
    public void eat() { // 必须要重写接口中的所有方法
        System.out.println("猫吃鱼");
    }
}

梳理一下接口和类之间的各种关系:

  • 类和类: extends, 单一继承,多层继承
  • 类和接口: implements, 多实现(可以实现多个接口)
  • 接口和接口: extends,接口之间不可以实现只可以继承,同时也是单一继承,多层继承

接口的优点:

  • 一个类可以实现多个接口,解决了extends的单一性问题
  • 接口的所有成员都是public,可以对外提供一套规范
  • 降低了程序的耦合性

接口和抽象类的比较:

  • 相似点:
    • 都是不断的抽取出抽象的概念
  • 不同点:
    • 抽象类也是类,只能单一继承,而类可以实现多个接口
    • 成员方面,抽象类可以有常量和成员变量,但是接口只能有变量。抽象类还可以有非抽象方法,接口必须都是抽象方法并且修饰符默认为public abstract
    • 构造方法方面,抽象类是有构造函数的,但是接口没有构造函数

匿名对象

没有变量引用的对象

public class AnooymousObjDemo {
    public static void main(String[] args) {
        new Test("DeeJay").show(); // name: DeeJay
    }
}

class Test {
    String name;
    public Test(String name) {
        this.name = name;
    }
    public void show() {
        System.out.println("name: "+ name);
    }
}

一般当方法只调用一次的时候可以使用匿名对象。

final关键字

修饰符,可以修饰类,成员方法,以及成员变量

  • final修饰的类不可以被继承。
  • final修饰的成员方法不可以被重写
  • final修饰的成员变量不可以被赋值,哪怕赋的是原值。即是常量

另外自定义常量必须初始化,可以直接赋值,或者在构造方法中初始化都可。


public class FinalDemo {
    public static void main(String[] args) {
        Animal3 a3 = new Animal3();
        // a3.num = 10; // the final fields cannot be assigned  final修饰的成员变量不可以被赋值,哪怕赋的是原值。即是常量,一般命名使用大写。
    }
}

final class Animal {
    public void eat(){
        System.out.println("eating");
    }
}

// class Dog extends Animal {} // cannot subclass the final class   final修饰的类不可以被继承

class Animal2 {
    final public void eat(){ // final修饰成员方法
        System.out.println("eating");
    }
}
class Dog2 extends Animal2 {
    // public void eat() {} // cannot Override the final method  final修饰的成员方法不可以被重写
}

class Animal3 {
    final int num = 3;
}

多态

多态的前提:

  • 子父类的继承关系
  • 方法的重写
  • 父类引用指向子类对象 Dad d = new Child();

多态的成员特点:

  • 成员变量:成员变量在继承时没有重写的概念,也没有动态绑定的概念,所以运行时指向的是父类的成员变量。
  • 成员方法: 成员方法运行时,指向的是子类中被重写的方法
  • 静态方法: 调用的静态方法是父类的静态方法,因为static是跟着类型走的。

public class PolymorphismDemo {
    public static void main(String[] args) {
        Dad d = new Child();
        System.out.println(d.num); // 20 成员变量在继承时没有重写的概念,也没有动态绑定的概念。 所以这边的是父类的成员变量。
        d.show(); // child 成员方法运行时,指向的是子类中被重写的方法
        d.show2(); // dad static  d变量类型还是Dad型,所以调用的静态方法是Dad类的静态方法
    }
}

class Dad {
    int num = 20;
    public void show() {
        System.out.println("dad");
    }
    public static void show2() {
        System.out.println("dad static");
    }
}
class Child extends Dad {
    int num = 10;
    public void show() {
        System.out.println("child");
    }
    public static void show2() {
        System.out.println("child static");
    }
}

多态中的类型转换:

public class PolymorphismDemo {
    public static void main(String[] args) {
        //向下转型
        Dad2 d2 = new Child2();
        Child2 c2 = (Child2)d2; // 强制的进行了类型转换  从Dad2类型到了Child2类型。 向下转型
        c2.show(); // child 向下转型了之后就可以调用子类的方法
    }
}

// 关于多态中引用类型的转换  向上转换Dad d = new Child(); 已经隐式的进行了转换,从小到大向上转换即从子类型到父类型
// 对于向下转换即从父类转到子类,一般是由于想访问子类中特有的成员才执行的,需要强制进行类型转换
class Dad2 {}
class Child2 extends Dad2{
    public void show() {
        System.out.println("child");
    }
}

多态的优缺点:

缺点:

  • 无法直接访问子类特有的成员,如果非要访问,需要向下转型
    优点:
  • 提高可维护性和可扩展性

对于提高可扩展性,来写一个例子:

package com.polymorphism;

public class PolymorphismDemo {
    public static void main(String[] args) {
        Factory f = new Factory();
        f.createPhone(new MiPhone());
        f.createPhone(new MeiZuPhone());
    }
}

class Factory {
    public void createPhone(MiPhone mp) { 
        System.out.println("create phone");
        mp.call();
    }
    
    public void createPhone(MeiZuPhone mzp) { // 随着手机类越来越多    Factory类中要一直新增createPhone的重载方法
        System.out.println("create phone");
        mzp.call();
    }
}

class MiPhone {
    public void call() {
        System.out.println("MI phone is calling");
    }
}

class MeiZuPhone {
    public void call () {
        System.out.println("MeiZuPhone is calling");
    }
}

上面这个例子中,Factory类中的方法随着手机类的增多,被迫要一次次的新增一个重载的方法。

我们可以使用多态来改写这个例子:

package com.polymorphism;

public class PolymorphismDemo2 {
    public static void main(String[] args) {
        Factory2 f2 = new Factory2();
        f2.createPhone(new MiPhone2());
        f2.createPhone(new MeiZuPhone2());
    }
}

class Factory2 {
    public void createPhone(Phone p) {  // 不需要每新增一个手机类就实现一次重载 
        System.out.println("create phone");
        p.call();
    }
}

interface Phone { // 创建一个公共的接口  让其他的手机类来实现这个接口   在Factory2的方法中就可以只传人实现了这个Phone 类型的变量
    public abstract void call(); 
}

class MiPhone2 implements Phone { // 实现公共接口   具体的方法在类内部进行定义
    public void call() {
        System.out.println("MI phone is calling");
    }
}

class MeiZuPhone2 implements Phone { // 实现公共接口   具体的方法在类内部进行定义
    public void call () {
        System.out.println("MeiZuPhone is calling");
    }
}

在改进的例子中,我们写了一个公共的接口Phone,所有的手机类都来实现这个接口,最后在Factory2中的方法中,我们只需要传入Phone p的参数就可以了,不用关心它具体是什么类型,从而就避免了一直重载方法的尴尬。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,542评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,596评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,021评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,682评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,792评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,985评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,107评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,845评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,299评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,612评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,747评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,441评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,072评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,828评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,069评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,545评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,658评论 2 350

推荐阅读更多精彩内容

  • 1 面向对象No6 面向对象 OO Object Oriented 编程时以对象为单元,封装数据和逻辑,以此提...
    征程_Journey阅读 1,132评论 0 2
  • 面向对象笔记 一、 对象在内存中的存放方法以及被调用过程 class文件首先被加载到方法区中的class文件内容区...
    VictorBXv阅读 460评论 0 2
  • java继承 继承的概念 继承是java面向对象编程技术的一块基石,因为它允许创建分等级层次的类。 继承就是子类继...
    863cda997e42阅读 661评论 0 1
  • 2018-5-8 下班和同事的约饭 大家一直七嘴八舌的开始吐槽 抱怨 我是一个不善言辞 也不喜欢群体活动的人 面对...
    臻悉阅读 144评论 0 0
  • 一枚令无数江湖人垂涎的令牌,一尾让无数人闻风丧胆的天煞琴,一把斩妖除魔的旷世神剑,一段恩怨情仇交织的江湖故事…… ...
    MJ老段阅读 664评论 8 15