面向对象程序设计

OOP

【面向对象程序设计】(OOP)与【面向过程程序设计】在思维方式上存在着很大的差别。【面向过程程序设计】中,算法是第一位的,数据结构是第二位的,这就明确地表述了程序员的工作方式。首先要确定如何操作数据,然后再决定如何组织数据,以便于数据操作。而【面向对象程序设计】却调换了这个次序,【面向对象程序设计】将数据放在第一位,然后再考虑操作数据的算法。

对于一些规模较小的问题,将问题分解为过程的开发方式比较理想。而面向对象更加适用于解决规模较大的问题。

面向对象程序设计是一种编程范式或编程风格。面向对象的程序是由类和对象组成的(以类和对象作为组织代码的基本单元),并将封装、抽象、继承、多态这四个特性,作为程序设计和实现的基础。

面向对象程序设计语言是【支持类和对象的语法机制。并有现成的语法机制,能方便地实现 OOP 的四大特性(封装、抽象、继承、多态)】的编程语言。

OOP 的四大特性

对于 OOP 的四大特性,我们需要知道每一个特性的如下知识:

  • xxx 特性的含义
  • 为了实现 xxx 特性,需要程序设计语言提供一定的语法机制来支持。对于这四大特性,尽管大部分面向对象程序设计语言都提供了相应的语法机制来支持,但不同的编程语言实现这四大特性的语法机制可能会有所不同。
  • xxx 特性存在的意义、好处

封装

封装(encapsulation)也被称为数据隐藏、数据访问保护。从形式上看,封装就是将数据和行为组合在一起中,并对对象的使用者隐藏数据的实现方式。

对象中的数据被称为实例域(instance field),操作数据的过程被称为方法(method)。对于每个特定的类实例(对象)都有一组特定的实例域值。这些值的集合就是这个对象的当前状态(state)。

实现封装的关键在于绝对不能让类中的方法直接地访问其他类的实例域。程序仅通过对象的方法与对象数据进行交互。封装给对象赋予了 “黑盒” 特征,这是提高重用性和可靠性的关键。这意味着一个类可以全面地改变存储数据的方式,只要仍旧使用同样的方法操作数据,其他对象就不会知道或介意所发生的变化。


为了实现封装这个特性,需要程序设计语言提供一定的语法机制来支持。这个语法机制就是访问权限控制(访问修饰符:public、protected、private、default)。

Java 中,封装就意味着所有的实例域都带有 private 访问修饰符(私有的实例域),并提供带有 public 访问修饰符的域访问器方法和域更改器方法(公共的操作方法)。

如果实例域带有 public 访问修饰符,这就破坏了封装性。因为 public 实例域允许程序中的任何方法对其进行读取和修改。

如果域访问器方法、域更改器方法直接返回了一个可变对象的引用,这就破坏了封装性。在 Employee 类中就违反了这个设计原则,其中的 getHireDay() 方法返回了一个 Date 类对象。Date 类有一个更改器方法 setTime(),可以使用 setTime() 这个方法设置毫秒数。Date 对象是可变的,这一点就破坏了封装性。对 d 调用更改器方法就可以自动地改变这个雇员对象的私有状态。

如果域访问器方法、域更改器方法需要返回一个可变对象的引用,应该首先对对象进行克隆(clone)。

对象 clone 指的是:存放在另一个位置上的对象副本。

class Employee {
    private Date hireDay;

    public Date getHireDay() {
        return hireDay; // Bad
    }
    // ...
}

Employee harry = . .
Date d = harry.getHireDay();
double tenYearsInMilliSeconds = 10 * 365.25 * 24 * 60 * 60 * 1000;
d.setTime(d.getTime() - (long) tenYearsInMilliSeconds);
// let's give Harry ten years of added seniority

// 修改后的代码
class Employee {
    public Date getHireDay() {
        return (Date) hireDay.clone(); // Ok
    }
    // ...
}

封装存在的意义、封装的好处:程序仅通过对象的方法与对象数据进行交互

  • 保护对象数据不被随意修改。
  • 可以改变类的内部实现,除了该类的方法之外,不会影响其他的代码。
  • 更改器方法可以执行错误检查,而直接对实例域进行赋值将不会进行这些处理。例如,setSalary 方法可以检查薪水是否小于 0。

抽象

封装主要讲的是如何隐藏数据、数据访问保护,而抽象讲的是如何隐藏方法的具体实现,让方法的调用者只需要关心方法提供了哪些功能,并不需要知道这些功能是如何实现的。


我们可以借助程序设计语言提供的接口类(比如 Java 中的 interface 关键字语法)或者抽象类(比如 Java 中的 abstract 关键字语法)这两种语法机制,来实现抽象这一特性。

实际上,抽象这个特性是非常容易实现的,并不需要非得依靠接口类或者抽象类这些语法机制来支持。换句话说,并不是说一定要为实现类抽象出接口类,才叫作抽象。即便不编写接口类,单纯的实现类本身就满足抽象特性。

之所以这么说,那是因为类的方法是通过程序设计语言中的 “函数” 这一语法机制实现的。通过函数包裹具体的实现逻辑,这本身就是一种抽象。调用者在调用函数的时候,并不需要去研究函数内部的实现逻辑,只需要通过函数的命名、注释或者文档,了解该函数提供了什么功能,就可以直接调用了。比如,我们在使用 C 语言的 malloc() 函数的时候,并不需要了解它的底层代码是怎么实现的。


抽象存在的意义、抽象的好处:

  • 一方面,抽象提高了代码的可扩展性、可维护性,修改实现不需要改变定义,减少了代码的改动范围;
  • 另一方面,抽象是处理复杂系统的有效手段,抽象能有效地过滤掉不必要关注的信息。

继承

继承(inheritance)即 “is-a” 关系,是一种用于表示特殊与一般关系的。

例如,RushOrder 类由 Order 类继承而来。在具有特殊性的 RushOrder 类中包含了一些用于优先处理的特殊方法,以及一个计算运费的不同方法;而其他的方法,如添加商品、生成账单等都是从 Order 类继承来的。

利用继承,人们可以基于已存在的类构造一个新类。继承已存在的类就是复用(继承)这些类的方法和域。在此基础上,还可以添加一些新的方法和域,以满足新的需求。

从继承关系上来讲,继承可以分为单继承和多继承。有些程序设计语言只支持单继承,不支持多重继承,比如 Java、PHP、C#、Ruby 等,而有些程序设计语言既支持单继承,也支持多继承,比如 C++、Python、Perl 等。

  • 单继承表示一个子类只能继承一个父类;
  • 多继承表示一个子类可以继承多个父类。

为了实现继承这个特性,需要程序设计语言提供一定的语法机制来支持。比如 Java 使用 extends 关键字来实现继承,C++ 使用冒号来实现继承(class B : public A),Python 使用 parentheses() 来实现继承,Ruby 使用 < 来实现继承。


继承存在的意义、继承的好处:继承的一个最大好处就是代码复用。假如两个类有一些相同的属性和方法,我们就可以将这些相同的部分,抽取到基类中,让两个子类继承基类。这样,两个子类就可以重用基类中的代码,避免代码重复写多遍。

不过,代码复用这个好处也并不是继承所独有的,我们也可以通过其他的方式来解决代码复用的问题,比如利用组合关系。

过度的使用继承,继承的层次过深、过复杂,就会导致代码的可读性、可维护性变差。

  • 可读性变差的原因:为了了解一个类的功能,我们不仅需要查看这个类的代码,还需要按照继承关系一层一层地往上查看“父类、父类的父类……”的代码。
  • 可维护性变差的原因:子类和父类高度耦合,修改父类的代码,会直接影响到子类。

多态

一个对象变量可以指向多种实际类型的现象被称为多态(polymorphism)。在运行时自动地选择调用哪个方法的现象被称为动态绑定(dynamic binding)。


为了实现多态这个特性,需要程序设计语言提供一定的语法机制来支持。

  • 第一个语法机制是:程序设计语言要支持继承;
  • 第二个语法机制是:程序设计语言要支持父类的对象变量可以引用子类对象;
  • 第三个语法机制是:程序设计语言要支持方法的重写(override)。

在 Java 程序设计语言中,对象变量是多态的。一个父类的对象变量既可以引用一个父类的对象,也可以引用一个子类的对象。

一个 Employee 变量既可以引用一个 Employee 类的对象,也可以引用一个 Employee 类的任何一个子类的对象(例如, Manager、Executive、 Secretary 等)。

对于多态特性的实现方式,除了利用 “继承加方法重写” 这种实现方式之外,还有其他两种比较常见的的实现方式,一种是利用接口类语法,另一种是利用 duck-typing 语法。不过,并不是每种程序设计语言都支持接口类或者 duck-typing 这两种语法机制,比如 C++ 就不支持接口类语法,而 duck-typing 只有一些动态语言才支持,比如 Python、JavaScript 等。

  • 接口类语法:一个对象变量(接口类)可以指向多种实际类型(实现类)
  • duck-typing 语法:duck-typing 可以这样表述:“如果看起来像鸭子,叫起来像鸭子,那么它一定是鸭子”。

多态存在的意义、多态的好处:

  • 多态的好处是,我们可以在一个比较稳定的、抽象的层面上编程,而不被更加具体的、易变的实现细节干扰。
  • 多态也是很多设计模式、设计原则、编程技巧的代码实现基础,比如策略模式、基于接口而非实现编程、依赖倒置原则、里式替换原则、利用多态去掉冗长的 if-else 语句等。

参考资料

《Java核心技术卷一:基础知识》(第10版)

04 | 理论一:当谈论面向对象的时候,我们到底在谈论什么?-极客时间 (geekbang.org)

05 | 理论二:封装、抽象、继承、多态分别可以解决哪些编程问题? (geekbang.org)

原文链接:https://www.cnblogs.com/feiyu2/p/17312680.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:面向对象程序设计 - Python技术站

(0)
上一篇 2023年4月18日
下一篇 2023年4月18日

相关文章

  • 详解Java中JSON数据的生成与解析

    详解Java中JSON数据的生成与解析 JSON数据格式现在已经成为了网络数据交换的标准之一,Java作为一门主流语言,当然也提供了很好的生成和解析JSON数据的库。下面我们来详细讲解如何在Java中生成和解析JSON数据。 JSON数据的生成 Java提供了许多第三方库用于生成JSON数据,其中Jackson库是比较常用的一种,我们就以这个库为例来讲解如何…

    Java 2023年5月26日
    00
  • Spring Boot security 默认拦截静态资源的解决方法

    Spring Boot Security默认会拦截所有请求,包括静态资源文件。这样会导致我们在访问静态资源时收到403(Forbidden)错误的响应。下面将介绍解决这个问题的方法。 第一种解决方法 第一种解决方法是在配置类上添加注解,忽略静态资源的拦截。 @Configuration @EnableWebSecurity public class Secu…

    Java 2023年6月3日
    00
  • Java的Struts框架报错“ActionForwardNotFoundException”的原因与解决办法

    当使用Java的Struts框架时,可能会遇到“ActionForwardNotFoundException”错误。这个错误通常由以下原因之一起: 配置错误:如果ActionForward配置不正确可能会出现。在这种情况下,需要检查ActionForward配置以解决此问题。 URL路径问题:如果URL路径不正确,则可能会出现此。在种情况下,需要检查URL路…

    Java 2023年5月5日
    00
  • 扫微信小程序码实现网站登陆实现解析

    首先,我们需要了解微信小程序码和网站的登陆原理: 微信小程序码是一种二维码,可以直接扫描进入微信小程序。在网站中使用微信小程序码登录,需要先在微信公众平台上绑定网站的 appid,并在小程序代码中引入网站的登录页面。 网站的登陆原理,一般都是使用账号密码等安全信息进行验证,并在验证成功后向用户发放 token,以标识用户的身份。在这个过程中,需要涉及到加密与…

    Java 2023年5月23日
    00
  • SSH框架网上商城项目第23战之在线支付功能实现

    SSH框架网上商城项目第23战之在线支付功能实现 1. 确认在线支付接口 首先,在实现在线支付功能之前,我们需要确认使用哪种在线支付接口,比如使用支付宝、微信等。针对每种接口,都有对应的API文档可以供开发人员参考。这里以使用支付宝支付为例进行说明。 2. 导入支付宝SDK 在使用支付宝API之前,我们需要先在项目中导入支付宝SDK。具体操作步骤如下: 访问…

    Java 2023年6月16日
    00
  • JVM工作原理和工作流程简述

    JVM工作原理和工作流程简述 JVM是Java虚拟机的缩写,是一种Java应用程序的运行环境。JVM的主要作用是将编写好的Java程序分解成字节码文件,然后在JVM中解释执行这些字节码,最终将结果输出。JVM具有跨平台、可移植、安全、动态性等优点,广泛应用于计算机领域。下面将对JVM工作原理和工作流程进行简述。 JVM工作原理 JVM的工作原理主要包括以下几…

    Java 2023年5月26日
    00
  • Java开启/关闭tomcat服务器的方法

    下面是Java开启/关闭tomcat服务器的方法的完整攻略: 确认Tomcat安装目录 在进行Tomcat服务器的开启和关闭前,我们要先确认Tomcat的安装目录。通常情况下,Tomcat会默认安装在/usr/local/tomcat目录下。如果您的Tomcat安装目录与该路径不同,需要将其替换为正确的目录。 开启Tomcat服务器 我们可以通过以下步骤开启…

    Java 2023年5月19日
    00
  • MT6589平台通话录音时播放提示音给对方功能的具体实现

    要实现“MT6589平台通话录音时播放提示音给对方功能”,需要在两个方面进行修改: 修改系统代码,使得当调用通话录音时,系统能够在录音开始时往话筒播放提示音; 修改通话录音应用程序的源代码,使得当开始录音时,能够调用系统接口往话筒播放提示音。 下面将具体介绍实现这一功能的步骤和示例: 步骤一:修改系统代码 打开系统源代码,找到通话录音相关的文件,例如Audi…

    Java 2023年5月23日
    00
合作推广
合作推广
分享本页
返回顶部