面向对象编程三⼤特性 --封装、继承、多态

封装

把客观事物封装成抽象的类,并且类可以把自己的数据和方法只让可信的类或者对象操作,对不可信的进行信息隐藏。封装是面向对象的特征之一,是对象和类概念的主要特性。

通俗的说,一个类就是一个封装了数据以及操作这些数据的代码的逻辑实体。在一个对象内部,某些代码或某些数据可以是私有的,不能被外界访问。

通过这种方式,对象对内部数据提供了不同级别的保护,以防止程序中无关的部分意外的改变或错误的使用了对象的私有部分。但是如果⼀个类没有提供给外界访问的⽅法,那么这个类也没有什么意义了。

我们来看一个常见的 类,比如:Student

public class Student implements Serializable {

    private Long id;
    private String name;

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

}
复制代码

将对象中的成员变量进行私有化,外部程序是无法访问的。对外提供了访问的方式,就是set和get方法。 而对于这样一个实体对象,外部程序只有赋值和获取值的权限,是无法对内部进行修改

继承

继承 就是子类继承父类的特征和行为,使得子类对象(实例)具有父类的实例域和方法,或子类从父类继承方法,使得子类具有父类相同的行为。 在 Java 中通过 extends 关键字可以申明一个类是从另外一个类继承而来的,一般形式如下:

class 父类 {
}

class 子类 extends 父类 {
}
复制代码

继承概念的实现方式有二类:实现继承接口继承

实现继承是指直接使用基类的属性和方法而无需额外编码的能力

接口继承是指仅使用属性和方法的名称、但是子类必须提供实现的能力

一般我们继承基本类和抽象类用 extends 关键字,实现接口类的继承用 implements 关键字。

注意点:

通过继承创建的新类称为“子类”或“派生类”,被继承的类称为“基类”、“父类”或“超类”。

继承的过程,就是从一般到特殊的过程。要实现继承,可以通过“继承”(Inheritance)和“组合”(Composition)来实现。

子类可以拥有父类的属性和方法。

子类可以拥有自己的属性和方法, 即⼦类可以对⽗类进⾏扩展。

子类可以重写覆盖父类的方法。

JAVA 只支持单继承,即一个子类只允许有一个父类,但是可以实现多级继承,及子类拥有唯一的父类,而父类还可以再继承。

使用implements关键字可以变相的使java具有多继承的特性,使用范围为类继承接口的情况,可以同时继承多个接口(接口跟接口之间采用逗号分隔)。

# implements 关键字

public interface A {
    public void eat();
    public void sleep();
}

public interface B {
    public void show();
}

public class C implements A,B {
}
复制代码

值得留意的是: 关于父类私有属性和私有方法的继承 的讨论 这个网上 有大量的争论,我这边以Java官方文档为准: With the use of the extends keyword, the subclasses will be able to inherit all the properties of the superclass except for the private properties of the superclass. 子类不能继承父类的私有属性(事实),但是如果子类中公有的方法影响到了父类私有属性,那么私有属性是能够被子类使用的。

官方文档 明确说明: private和final不被继承,但从内存的角度看的话:父类private属性是会存在于子类对象中的。

通过继承的方法(比如,public方法)可以访问到父类的private属性

如果子类中存在与父类private方法签名相同的方法,其实相当于覆盖

个人觉得文章里的一句话很赞,我们不可能完全继承父母的一切(如性格等),但是父母的一些无法继承的东西却仍会深刻的影响着我们。

多态

同一个行为具有多个不同表现形式或形态的能力就是 多态。网上的争论很多,笔者个人认同网上的这个观点:重载也是多态的一种表现,不过多态主要指运行时多态

Java 多态可以分为 重载式多态重写式多态:

-重载式多态,也叫编译时多态。编译时多态是静态的,主要是指方法的重载,它是根据参数列表的不同来区分不同的方法。通过编译之后会变成两个不同的方法,在运行时谈不上多态。也就是说这种多态再编译时已经确定好了。

-重写式多态,也叫运行时多态。运行时多态是动态的,主要指继承父类和实现接口时,可使用父类引用指向子类对象实现。这个就是大家通常所说的多态性

这种多态通过动态绑定(dynamic binding)技术来实现,是指在执行期间判断所引用对象的实际类型,根据其实际的类型调用其相应的方法。也就是说,只有程序运行起来,你才知道调用的是哪个子类的方法。 这种多态可通过函数的重写以及向上转型来实现。

多态存在的三个必要条件:

  1. 继承
  2. 重写
  3. 父类引用指向子类对象:Parent p = new Child();

我们一起来看个例子,仔细品读代码,就明白了:

@SpringBootTest
class Demo2021ApplicationTests {

    class Animal {
        public void eat(){
            System.out.println("动物吃饭!");
        }
        public void work(){
            System.out.println("动物可以帮助人类干活!");
        }
    }

    class Cat extends Animal {
        public void eat() {
            System.out.println("吃鱼");
        }
        public void sleep() {
            System.out.println("猫会睡懒觉");
        }
    }

    class Dog extends Animal {
        public void eat() {
            System.out.println("吃骨头");
        }
    }

    @Test
    void contextLoads() {
        //part1
        Cat cat_ = new Cat();
        cat_.eat();
        cat_.sleep();
        cat_.work();

        //part2
        Animal cat=new Cat();
        cat.eat();
        cat.work();
        cat.sleep();//此处编译会报错。

        Animal dog=new Dog();
        dog.eat();//结果为:吃骨头。此处调用子类的同名方法。

        //part3
        //如果想要调用父类中没有的方法,则要向下转型,这个非常像"强转"
        Cat cat222 = (Cat)cat;        // 向下转型(注意,如果是(Cat)dog 则会报错)
        cat222.sleep();        //结果为:猫会睡懒觉。 可以调用 Cat 的 sleep()
    }

}
复制代码

我们来看上面part1部分:

Cat cat_ = new Cat();
cat_.eat();
cat_.sleep();
cat_.work();
复制代码

结果:

吃鱼 猫会睡懒觉。 动物可以帮助人类干活!

cat_.work(); 这处继承了父类Animal的方法,还是很好理解的 我们接着看part2:

Animal cat=new Cat();
cat.eat();
cat.work();
cat.sleep();//此处编译会报错。

Animal dog=new Dog();
dog.eat();//结果为:吃骨头。此处调用子类的同名方法。
复制代码

这块就比较特殊了,我们一句句来看

Animal cat=new Cat(); 像这种这个 父类引用指向子类对象,这种现象叫做:"向上转型",也被称为多态的引用

cat.sleep();这句 编译器会提示 编译报错。 表明:当我们当子类的对象作为父类的引用使用时,只能访问子类中和父类中都有的方法,而无法去访问子类中特有的方法

值得注意的是:向上转型是安全的。但是缺点是:一旦向上转型,子类会丢失的子类的扩展方法,其实就是 子类中原本特有的方法就不能再被调用了。所以cat.sleep()这句会编译报错。

cat.eat();这句的结果打印:吃鱼。程序这块调用我们Cat定义的方法。

cat.work();这句的结果打印:动物可以帮助人类干活! 我们上面Cat类没有定义work方法,但是却使用了父类的方法,这是不是很神奇。其实此处调的是父类的同名方法

Animal dog=new Dog();dog.eat();这句的结果打印为:吃骨头。此处调用子类的同名方法。

由此我们可以知道当发生向上转型,去调用方法时,首先检查父类中是否有该方法,如果没有,则编译错误;如果有,再去调用子类的同名方法。如果子类没有同名方法,会再次去调父类中的该方法

我们现在知道了 向上转型时会丢失子类的扩展方法,哎,但我们就是想找回来,这可咋办? 向下转型可以帮助我们,找回曾经失去的

我们来看part3:

    Cat cat_real = (Cat)cat;  //注意 此处的cat 对应上面父类Animal,可不是子类
    cat_real.sleep(); 
复制代码

Cat cat = (Cat)cat; cat222.sleep(); 这个向下转型非常像"强转"。

打印的结果:猫会睡懒觉。此处又能调用了 子类Cat 的 sleep()方法了。

一道简单的面试题

我们再来看一道有意思的题,来强化理解

public class Main {

    static class Animal{
        int weight = 10;

        public void print() {
            System.out.println("this Animal Print:" + weight);
        }

        public Animal() {
            print();
        }
    }

    static class Dog extends Animal {
        int weight = 20;

        @Override
        public void print() {
            System.out.println("this Dog Print:" + weight);
        }

        public Dog() {
            print();
        }
    }

    public static void main(String[] args) {
        Dog dog = new Dog();

        System.out.println("---------------");
        Animal dog222 = new Dog();
        Dog dog333 =  (Dog)dog222;

        System.out.println("---------------");
        Dog dog444 = (Dog)new Animal();

    }
}
复制代码

执行结果:

this Dog Print:0 this Dog Print:20


this Dog Print:0

this Dog Print:20


this Animal Print:10 Exception in thread "main" java.lang.ClassCastException: com.zj.Main Animalcannotbecasttocom.zj.MainDog at com.zj.Main.main(Main.java:15)

做对了嘛,不对的话,复制代码去idea中debug看看 [图片上传失败...(image-f7f8e1-1666835336011)]

我们先看第一部分

Dog dog = new Dog();

程序内部的执行顺序:

  1. 先 初始化 父类Animal 的属性 int weight=10
  2. 然后 调用父类Animal的构造方法,执行print()
  3. 实际调用子类Dog的print()方法,打印:this Dog Print:0,由于此时的子类属性weight 并未初始化
  4. 初始化 子类Dog 的属性 int weight=20
  5. 调用 子类Dog的构造方法,执行print()
  6. 实际调用当前类的print()方法,打印this Dog Print:20

其中有几处我们需要注意一下:实例化子类dog,程序会去默认优先实例化父类,即子类实例化时会隐式传递Dog的this调用父类构造器进行初始化工作,这个和JVM的双亲委派机制有关,这里就不展开讲了,先挖个坑,以后再来填🙃。

当程序调用父类Animal的构造方法时,会隐式传递Dog的this,类似于:

Dog dog = new Dog(this);

static class Animal{
  public Animal(this) {
      print(this);//子类又把print()给重写了
  }
}

static class Dog extends Animal {
  public Dog(this) {
      print(this);//此时子类的 属int weight 被没有初始化默认为0
  }
}
复制代码

这块其实和JVM的虚方法表有关,这又是一个大坑,以后慢慢填🙃。

我们接着看第2部分 Animal dog222 = new Dog();这句是向上转型,程序加载顺序和第一部分Dog dog = new Dog();是一样的,都是实例化类的过程 Dog dog333 = (Dog)dog222;这个是向下转型,并没有调用类构造器,这块等会和第3部分结合讲

最后我们来看下第3部分Dog dog444 = (Dog)new Animal();这句先实例化Andimal类,它没有父类,就直接实例化当前类,打印this Animal Print:10。然后(Dog)表示向下转型,但是为啥运行会报ClassCastException 异常呢?且第2部分Dog dog333 = (Dog)dog222;却没有问题?

我们可以发现,向下转型可以让子类找回其独有的方法 但是向下转型是不安全的,实现 向下转型 前需要先实现 向上转型。

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

推荐阅读更多精彩内容