面向对象

面向对象编程

面向对象编程(Object Oriented Programming, OOP)是一种编程范式或编程风格,它以类或对象作为组织代码的基本单元,并以封装、继承、多态这三个特性作为代码设计和实现的基石。

面向对象的类是描述了一组有相同特征(属性)和相同行为(方法)的一组对象的集合;对象是类的一个实例,拥有自己的状态和行为。

面向对象编程语言

面向对象编程语言(Object Oriented Programming Language, OOPL)是支持以类或对象的语法机制,并有现成的语法机制,能方便地实现面向对象编程的三大特性(封装、继承、多态)的编程语言。

面向对象编程设计和分析

面向对象分析(Object Oriented Analysis, OOA)就是要搞清楚是什么、为什么要做。

面向对象设计(Object Oriented Design, OOD)就是要搞清楚由谁来做、什么时候做、在哪里做、怎么做、做到怎样的程度。

设计和分析就是一个将想法付诸于实际的过程,因此,其中的每一步都非常重要,影响到程序这项工程的维护。

面向对象的优点

面向对象程序设计有以下优点:

  • 可重用性:代码重复可用,减少代码量,提高开发效率
  • 可扩展性:新的功能可以很容易地加入到系统中来,便于软件的修改
  • 可管理性:能够将功能和数据结合,方便管理

核心特性

封装

封装的含义

封装也叫作信息隐藏或数据访问保护。详细地说,就是数据被保护在抽象数据类型的内部,尽可能对外隐藏内部的细节,只保留一些统一的方法供外部使用。

比如说,对于一个钱包类,里面有余额、币种这两个属性,通常是不允许外部直接更新余额或者直接更新币种,而是仿照现实交易的找补零钱的方式,对外提供一个找补零钱的方法,在这个方法中根据提供的参数来更新余额和币种,这样可以保证数据的一致性。

封装的优点

封装具有以下优点:

  • 提高了代码的安全性,阻止外部随意修改,避免造成数据不一致
  • 提高了代码的易用性,简化外部调用,便于扩展和协作
  • 提高了代码的可维护性,封装内部细节,方便修改内部代码

继承

继承的含义

继承指的是子类拥有父类的全部特征和行为,用来表示类之间 is-A 的关系。

比如说,汽车是一种交通工具,汽车会有交通工具的一些特性和功能,交通工具狭义上指一切人造的用于人类代步或运输的装置,汽车就属于这类工具中的一种。

单继承和多继承

从继承的多向性来讲,继承可分为两种模式:单继承和多继承。

单继承表示一个子类只能继承一个父类,多继承表示一个子类可以继承多个父类。从现实世界的角度上看,多继承更符合现实,比如说,猫既是哺乳动物,又是爬行动物。

但是,从软件开发的角度上看,单继承的优点在于层次结构清晰,设计上更容易把握;多继承可以让子类具备多个父类的特征,拥有更丰富的方法,但是多继承会出现菱形继承的问题。

简单地理解菱形继承就是,假设子类 B 和子类 C 都继承自父类 A,且都重写了父类 A 中的方法 func,而孙子类 D 同时继承了子类 B 和子类 C,对于方法 func 而言,孙子类 D 会出现歧义。

继承的优缺点

继承最大的好处就是代码复用,子类可以直接重用父类中的代码,避免代码重复写多遍。

但是过度地使用继承会导致代码可读性、可维护性变差,有可能出现“父类、父类的父类……”的代码。

通常,可以在层次简单、关系不复杂的时候使用继承,反之使用组合代替继承。

多态

多态的含义

多态指的是为不同数据类型的实体提供统一的接口,或使用一个单一的符号来表示多个不同的类型。

通过继承实现

多态可以通过继承的方式实现,子类继承父类之后,并重写了父类的方法,在初始化子类的对象时,可以将对象定义为父类的数据类型,这时的对象调用的会是重写后的子类方法。

如下述代码所示:

package cn.fatedeity.designpattern.polymorphism;

public class extendCase {
    private static void test(Base base) {
        System.out.println(base.getSize());
    }

    public static void main(String[] args) {
        Base baseAddOne = new BaseAddOne();
        // 1
        test(baseAddOne);

        Base baseAddTwo = new BaseAddTow();
        // 2
        test(baseAddTwo);
    }
}

class BaseAddTow extends Base {
    @Override
    public int getSize() {
        return this.size + 2;
    }
}

class BaseAddOne extends Base {
    @Override
    public int getSize() {
        return this.size + 1;
    }
}

class Base {
    protected int size = 0;

    public int getSize() {
        return size;
    }
}

通过接口实现

多态还可以通过接口的方式实现,当接口被实现之后,在初始化实现类的对象时,可以直接将这个对象定义为接口类型,这时的对象调用的会是实现类的方法。

如下述代码所示:

package cn.fatedeity.designpattern.polymorphism;

public class ImplementsCase {
    private static void test(InterfaceBase base) {
        System.out.println(base.toString());
    }

    public static void main(String[] args) {
        InterfaceBase interfaceOne = new InterfaceOne();
        // This is InterfaceOne
        test(interfaceOne);

        InterfaceBase interfaceTwo = new InterfaceTow();
        // This is InterfaceTwo
        test(interfaceTwo);
    }
}

class InterfaceTow implements InterfaceBase {
    @Override
    public String toString() {
        return "This is InterfaceTwo";
    }
}

class InterfaceOne implements InterfaceBase {
    @Override
    public String toString() {
        return "This is InterfaceOne";
    }
}

interface InterfaceBase {
    String toString();
}

通过鸭子类型实现

所谓的鸭子类型,指的是只关心事物的外部行为而非内部结构,即不关心对象是什么类型,只关心该对象是否拥有指定方法。

通过鸭子类型实现多态更加灵活,不需要类之间有继承、接口实现的关系,只需要它们同时定义了相同的方法即可。如下述的 Python 代码所示:

class Logger:
    def record(self):
        print('I write a log into file.')

class DB:
    def record(self):
        print('I insert data into db.')

def test(recorder):
    recorder.record()

def demo():
    logger = Logger()
    # I write a log into file.
    test(logger)

    db = DB()
    # I insert data into db.
    test(db)

多态的意义

对于第一个例子的代码,仅用一个 test() 方法即可测试 Base 类的子类,即使要新增一个 BaseAddThree 子类,同样不需要更改 test() 方法,仅需重写自己的 getSize() 方法即可,这里提高了代码的扩展性。

同样的,仅用一个 test() 即可完成所有的测试,而不需对每一个子类都写一遍测试代码,这里显然提高了代码的复用性。

除此之外,多态还是很多设计模式、设计原则、编程技巧的代码实现基础。

面向对象和面向过程

为什么使用面向对象而不是面向过程?

面向过程是一种流程化的思维模式,面向对象是一种自底向上的抽象化的思维模式。

相比之下,面向对象有以下优势:

  • 面向对象编程更加能够应对大规模复杂程序的开发,它提供了一种清晰的、模块化的代码组织方式
  • 面向对象编程的的三大特性提高了代码的易维护性、扩展性、复用性,并且大部分设计模式都以面向对象为基础
  • 面向对象编程语言更加人性化、更加高级、更加智能,面向过程的流程化是一种计算机思维方法,而面向对象的抽象化是一种人类思维方法

违反面向对象编程风格的典型代码设计

  • 滥用 setter() 方法和 getter() 方法使封装失去作用
  • 定义大而全的 Constants 类、Utils 类也破坏了封装特性
  • MVC 模式是基于贫血模型的开发模式,数据和操作分开,是彻底的面向过程编程风格
08-05 06:25