面向对象编程

328次阅读  |  发布于2年以前

本文主要分三部分:常见的编程范式、面向对象编程的基本特征、代码设计原则。

常见的编程范式

常见的编程范式有三种:面向对象编程、面向过程编程和函数式编程。

面向对象编程更适合构架大规模复杂应用,采用面向对象编程思路编写的代码更加容易扩展和维护。面向过程编程代码写起来更加容易,直接了当,但是代码的复用性和封装性比较差。函数式编程代码复用性、抽象级别更高、代码健壮稳定,但是二次修改比较困难。

Java是典型的面向对象开发编程语言,因为Java很好的支持类、继承、多态等特性。同时Java也可以实现面向过程编程和函数式编程。

Golang虽然没有类的概念但是通过struct和匿名字段也可以实现继承、封装、多态的特性,所以说Golang也支持面向对象编程,不过实现思路和传统的面向对象编程语言的思路有些差异。函数在Golang中属于一等公民,可以作为方法的参数和返回值所以Golang也非常适合函数式编程。

JavaScript非常适合函数式编程,因为:

  1. 函数在JavaScript中是一等公民,可以作为参数和返回值。
  2. 它是一门动态语言可以根据字符串快速定位到Object的一个函数属性。
  3. 支持闭包。

JavaScript也支持面向对象编程,因为:

  1. JavaScript的原型使它可以变相实现类的概念,也可以实现继承和多态。
  2. ES6标准支持class。
  3. TypeScript的出现使JavaScript书写起来更加自然地贴近面向对象开发。

我们举一个不太恰当的例子来理解三种编程范式,春晚有一个小品中说“把大象放进冰箱分几步?” 如果是面向过程编程,分三步:1.打开冰箱门。2.放进大象。3.关闭冰箱门。 如果是面向对象编程:先定义一个对象叫冰箱,再给冰箱“门”和“内容”属性,属性有两个值代表关闭状态和打开状态。门在打开状态时,可以给内容属性赋值“大象”。 如果是函数式编程:调用冰箱函数,返回一个往里放东西的函数。把大象传给返回的函数。

在实际开发中三种编程范式都有适合的场景。

面向对象编程的基本特征

三大特征:继承、封装、多态 封装:暴露有限的方法和属性,把逻辑收敛在内部。 继承:子类继承父类,可以复用父类中的代码。继承层级不易过深。 多态:父对象类型支持子对象的引用和传递。子对象可以重写父对象中的方法。

代码设计原则

常见的设计原则:

  1. 单一职责原则(Single Responsibility Principle)简称:SRP
  2. 开放封闭原则(Open Closed Principle)简称:OCP
  3. 里氏替换原则(Liskov Substitution Principle)简称:LSP
  4. 接口隔离原则(Interface Segregation Principle)简称:ISP
  5. 依赖倒置原则(Dependency Inversion Principle)简称:DIP
  6. 最少知识原则(The Least Knowledge Principle)简称:LKP,又叫迪米特原则
  7. 合成复用原则(Composite Reuse Principle)简称:CRP 其中,前五条又统称为'SOLID'原则。

单一职责原则

“A class or module should have a single responsibility.” 一个类或者模块只负责一个职责。 不要设计大而全的类,设计粒度小、功能单一的类。 并不是拆分得越细越好,没有明确的判断标准。需要对业务足够了解才能比较好的拆分。

开放封闭原则

“Software entities (modules, classes, functions, etc.) should be open for extension , but closed for modification.”

  1. 对扩展开放、对修改封闭。
  2. 对于新的功能尽量通过扩展类、模块、方法、属性来完成,而不是修改原有代码。
  3. 以最小的代码修改代价完成新功能的开发。举一个前端开发中的例子,比如说我们开发一个公共的弹框功能,第一版的需求是点击右上角的“关闭”按钮弹框会关闭,后来需求有变动,点击弹框之外的任何区域都需要关闭弹框,此时应该添加新的代码逻辑来处理关闭,而不是修改原来的close方法,在close方法中判断点击位置。

里氏替换原则

“Functions that use pointers of references to base classes must be able to use objects of derived classes without knowing it.”

子类能够替换父类出现的任何地方,并且能保证程序的逻辑行为正确。 这条原则的前提是已经有了继承关系。 举个例子,在父类A中有一个排序方法,调用之后会把传入的数组从小到大排序并返回,在子类B中重写了排序方法,会把传入的数组从大到小排序并返回。这明显违背了里氏替换原则。

接口隔离原则

“Clients should not be forced to depend upon interfaces that they do not use”。

使用多个隔离的接口,比使用单个接口要好。 目的是降低接口之间的耦合度。 这与在开发基础库时,避免过多的依赖导致技术绑架是异曲同工。此原则主要针对接口,和单一职责原则类似,提供了判断接口是否符合单一职责的判断标准。

依赖倒置原则

“High-level modules shouldn’t depend on low-level modules. Both modules should depend on abstractions. In addition, abstractions shouldn’t depend on details. Details depend on abstractions.”

高层模块(high-level modules)不要依赖低层模块(low-level)。高层模块和低层模块应该通过抽象(abstractions)来互相依赖。除此之外,抽象(abstractions)不要依赖具体实现细节(details),具体实现细节(details)依赖抽象(abstractions)。

所谓依赖倒置原则(Dependence Inversion Principle)就是要依赖于抽象,不要依赖于具体。简单的说就是要求对抽象进行编程,不要对实现进行编程,这样就降低了客户与实现模块间的耦合。

实现开闭原则的关键是抽象化,并且从抽象化导出具体化实现,如果说开闭原则是面向对象设计的目标的话,那么依赖倒转原则就是面向对象设计的主要手段。

最少知识原则

“Each unit should have only limited knowledge about other units: only units “closely” related to the current unit. Or: Each unit should only talk to its friends; Don’t talk to strangers.”

不该有直接依赖关系的类之间,不要有依赖;有依赖关系的类之间,尽量只依赖必要的接口(也就是定义中的“有限知识”)

合成复用原则

合成复用原则就是指在一个新的对象里通过关联关系(包括组合关系和聚合关系)来使用一些已有的对象,使之成为新对象的一部分;新对象通过委派调用已有对象的方法达到复用其已有功能的目的。

简言之:要尽量使用组合/聚合关系,少用继承。

总结

本文我们介绍了常见的编程范式、面向对象编程的基本特征和代码设计原则。这些知识是学习设计模式的基础,后面我们再写几篇文章介绍下常用的设计模式。

Copyright© 2013-2020

All Rights Reserved 京ICP备2023019179号-8