谈一谈基于python的面向对象编程基础


Posted in Python onMay 21, 2019

活在当下的程序员应该都听过“面向对象编程”一词,也经常有人问能不能用一句话解释下什么是“面向对象编程”,我们先来看看比较正式的说法。

把一组数据结构和处理它们的方法组成对象(object),把相同行为的对象归纳为类(class),通过类的封装(encapsulation)隐藏内部细节,通过继承(inheritance)实现类的特化(specialization)和泛化(generalization),通过多态(polymorphism)实现基于对象类型的动态分派。

这样一说是不是更不明白了。所以我们还是看看更通俗易懂的说法,下面这段内容来自于知乎。

谈一谈基于python的面向对象编程基础

说明: 以上的内容来自于网络,不代表作者本人的观点和看法,与作者本人立场无关,相关责任不由作者承担。

之前我们说过“程序是指令的集合”,我们在程序中书写的语句在执行时会变成一条或多条指令然后由CPU去执行。当然为了简化程序的设计,我们引入了函数的概念,把相对独立且经常重复使用的代码放置到函数中,在需要使用这些功能的时候只要调用函数即可;如果一个函数的功能过于复杂和臃肿,我们又可以进一步将函数继续切分为子函数来降低系统的复杂性。但是说了这么多,不知道大家是否发现,所谓编程就是程序员按照计算机的工作方式控制计算机完成各种任务。但是,计算机的工作方式与正常人类的思维模式是不同的,如果编程就必须得抛弃人类正常的思维方式去迎合计算机,编程的乐趣就少了很多,“每个人都应该学习编程”这样的豪言壮语就只能说说而已。当然,这些还不是最重要的,最重要的是当我们需要开发一个复杂的系统时,代码的复杂性会让开发和维护工作都变得举步维艰,所以在上世纪60年代末期,“软件危机”、“软件工程”等一系列的概念开始在行业中出现。

当然,程序员圈子内的人都知道,现实中并没有解决上面所说的这些问题的“银弹”,真正让软件开发者看到希望的是上世纪70年代诞生的Smalltalk编程语言中引入的面向对象的编程思想(面向对象编程的雏形可以追溯到更早期的Simula语言)。按照这种编程理念,程序中的数据和操作数据的函数是一个逻辑上的整体,我们称之为“对象”,而我们解决问题的方式就是创建出需要的对象并向对象发出各种各样的消息,多个对象的协同工作最终可以让我们构造出复杂的系统来解决现实中的问题。

说明: 当然面向对象也不是解决软件开发中所有问题的最后的“银弹”,所以今天的高级程序设计语言几乎都提供了对多种编程范式的支持,Python也不例外。

类和对象

简单的说,类是对象的蓝图和模板,而对象是类的实例。这个解释虽然有点像用概念在解释概念,但是从这句话我们至少可以看出,类是抽象的概念,而对象是具体的东西。在面向对象编程的世界中,一切皆为对象,对象都有属性和行为,每个对象都是独一无二的,而且对象一定属于某个类(型)。当我们把一大堆拥有共同特征的对象的静态特征(属性)和动态特征(行为)都抽取出来后,就可以定义出一个叫做“类”的东西。

谈一谈基于python的面向对象编程基础

定义类

在Python中可以使用class关键字定义类,然后在类中通过之前学习过的函数来定义方法,这样就可以将对象的动态特征描述出来,代码如下所示。

class Student(object):

 # __init__是一个特殊方法用于在创建对象时进行初始化操作
 # 通过这个方法我们可以为学生对象绑定name和age两个属性
 def __init__(self, name, age):
  self.name = name
  self.age = age

 def study(self, course_name):
  print('%s正在学习%s.' % (self.name, course_name))

 # PEP 8要求标识符的名字用全小写多个单词用下划线连接
 # 但是很多程序员和公司更倾向于使用驼峰命名法(驼峰标识)
 def watch_av(self):
  if self.age < 18:
   print('%s只能观看《熊出没》.' % self.name)
  else:
   print('%s正在观看岛国爱情动作片.' % self.name)

说明: 写在类中的函数,我们通常称之为(对象的)方法,这些方法就是对象可以接收的消息。

创建和使用对象

当我们定义好一个类之后,可以通过下面的方式来创建对象并给对象发消息。

def main():
 # 创建学生对象并指定姓名和年龄
 stu1 = Student('骆昊', 38)
 # 给对象发study消息
 stu1.study('Python程序设计')
 # 给对象发watch_av消息
 stu1.watch_av()
 stu2 = Student('王大锤', 15)
 stu2.study('思想品德')
 stu2.watch_av()


if __name__ == '__main__':
 main()

访问可见性问题

对于上面的代码,有C++、Java、C#等编程经验的程序员可能会问,我们给Student对象绑定的name和age属性到底具有怎样的访问权限(也称为可见性)。因为在很多面向对象编程语言中,我们通常会将对象的属性设置为私有的(private)或受保护的(protected),简单的说就是不允许外界访问,而对象的方法通常都是公开的(public),因为公开的方法就是对象能够接受的消息。在Python中,属性和方法的访问权限只有两种,也就是公开的和私有的,如果希望属性是私有的,在给属性命名时可以用两个下划线作为开头,下面的代码可以验证这一点。

class Test:

 def __init__(self, foo):
  self.__foo = foo

 def __bar(self):
  print(self.__foo)
  print('__bar')


def main():
 test = Test('hello')
 # AttributeError: 'Test' object has no attribute '__bar'
 test.__bar()
 # AttributeError: 'Test' object has no attribute '__foo'
 print(test.__foo)


if __name__ == "__main__":
 main()

但是,Python并没有从语法上严格保证私有属性或方法的私密性,它只是给私有的属性和方法换了一个名字来“妨碍”对它们的访问,事实上如果你知道更换名字的规则仍然可以访问到它们,下面的代码就可以验证这一点。之所以这样设定,可以用这样一句名言加以解释,就是“We are all consenting adults here”。因为绝大多数程序员都认为开放比封闭要好,而且程序员要自己为自己的行为负责。

class Test:

 def __init__(self, foo):
  self.__foo = foo

 def __bar(self):
  print(self.__foo)
  print('__bar')


def main():
 test = Test('hello')
 test._Test__bar()
 print(test._Test__foo)


if __name__ == "__main__":
 main()

在实际开发中,我们并不建议将属性设置为私有的,因为这会导致子类无法访问(后面会讲到)。所以大多数Python程序员会遵循一种命名惯例就是让属性名以单下划线开头来表示属性是受保护的,本类之外的代码在访问这样的属性时应该要保持慎重。这种做法并不是语法上的规则,单下划线开头的属性和方法外界仍然是可以访问的,所以更多的时候它是一种暗示或隐喻

面向对象的支柱

面向对象有三大支柱:封装、继承和多态。后面两个概念在下一个章节中进行详细的说明,这里我们先说一下什么是封装。我自己对封装的理解是“隐藏一切可以隐藏的实现细节,只向外界暴露(提供)简单的编程接口”。我们在类中定义的方法其实就是把数据和对数据的操作封装起来了,在我们创建了对象之后,只需要给对象发送一个消息(调用方法)就可以执行方法中的代码,也就是说我们只需要知道方法的名字和传入的参数(方法的外部视图),而不需要知道方法内部的实现细节(方法的内部视图)。

练习

练习1:定义一个类描述数字时钟

class Clock(object):
 """数字时钟"""

 def __init__(self, hour=0, minute=0, second=0):
  """初始化方法

  :param hour: 时
  :param minute: 分
  :param second: 秒
  """
  self._hour = hour
  self._minute = minute
  self._second = second

 def run(self):
  """走字"""
  self._second += 1
  if self._second == 60:
   self._second = 0
   self._minute += 1
   if self._minute == 60:
    self._minute = 0
    self._hour += 1
    if self._hour == 24:
     self._hour = 0

 def show(self):
  """显示时间"""
  return '%02d:%02d:%02d' % \
    (self._hour, self._minute, self._second)


def main():
 clock = Clock(23, 59, 58)
 while True:
  print(clock.show())
  sleep(1)
  clock.run()


if __name__ == '__main__':
 main()

练习2:定义一个类描述平面上的点并提供移动点和计算到另一个点距离的方法。

from math import sqrt


class Point(object):

 def __init__(self, x=0, y=0):
  """初始化方法
  
  :param x: 横坐标
  :param y: 纵坐标
  """
  self.x = x
  self.y = y

 def move_to(self, x, y):
  """移动到指定位置
  
  :param x: 新的横坐标
  "param y: 新的纵坐标
  """
  self.x = x
  self.y = y

 def move_by(self, dx, dy):
  """移动指定的增量
  
  :param dx: 横坐标的增量
  "param dy: 纵坐标的增量
  """
  self.x += dx
  self.y += dy

 def distance_to(self, other):
  """计算与另一个点的距离
  
  :param other: 另一个点
  """
  dx = self.x - other.x
  dy = self.y - other.y
  return sqrt(dx ** 2 + dy ** 2)

 def __str__(self):
  return '(%s, %s)' % (str(self.x), str(self.y))


def main():
 p1 = Point(3, 5)
 p2 = Point()
 print(p1)
 print(p2)
 p2.move_by(-1, 2)
 print(p2)
 print(p1.distance_to(p2))


if __name__ == '__main__':
 main()

说明: 本章中的插图来自于Grady Booch等著作的《面向对象分析与设计》一书,该书是讲解面向对象编程的经典著作,有兴趣的读者可以购买和阅读这本书来了解更多的面向对象的相关知识。

Python 相关文章推荐
python统计字符串中指定字符出现次数的方法
Apr 04 Python
举例讲解Python中的算数运算符的用法
May 13 Python
Python修改MP3文件的方法
Jun 15 Python
python检测是文件还是目录的方法
Jul 03 Python
Python数字图像处理之霍夫线变换实现详解
Jan 12 Python
Python3 修改默认环境的方法
Feb 16 Python
详解用python写一个抽奖程序
May 10 Python
python实现连连看辅助之图像识别延伸
Jul 17 Python
对Django url的几种使用方式详解
Aug 06 Python
python如何将多个PDF进行合并
Aug 13 Python
Python3之字节串bytes与字节数组bytearray的使用详解
Aug 27 Python
python中time、datetime模块的使用
Dec 14 Python
python字符串和常用数据结构知识总结
May 21 #Python
Opencv实现抠图背景图替换功能
May 21 #Python
python多进程读图提取特征存npy
May 21 #Python
Python中使用pypdf2合并、分割、加密pdf文件的代码详解
May 21 #Python
python+selenium实现简历自动刷新的示例代码
May 20 #Python
图文详解python安装Scrapy框架步骤
May 20 #Python
Python配置虚拟环境图文步骤
May 20 #Python
You might like
解决了Ajax、MySQL 和 Zend Framework 的乱码问题
2009/03/03 PHP
PHP中获取变量的变量名的一段代码的bug分析
2011/07/07 PHP
PHP备份数据库生成SQL文件并下载的函数代码
2012/02/05 PHP
PHP实现远程下载文件到本地
2015/05/17 PHP
详解PHP中的Traits
2015/07/29 PHP
php中文字符串截取多种方法汇总
2016/10/06 PHP
php导出csv文件,可导出前导0实例代码
2016/11/16 PHP
form表单传递数组数据、php脚本接收的实例
2017/02/09 PHP
Gambit vs CL BO3 第三场 2.13
2021/03/10 DOTA
Jquery.LazyLoad.js修正版下载,实现图片延迟加载插件
2011/03/12 Javascript
jquery获取radio值(单选组radio)
2014/10/16 Javascript
基于javascript实现按圆形排列DIV元素(一)
2016/12/02 Javascript
在页面中引入js的两种方法(推荐)
2017/08/29 Javascript
JS中关于正则的巧妙操作
2017/08/31 Javascript
Angular中封装fancyBox(图片预览)遇到问题小结
2017/09/01 Javascript
Angular浏览器插件Batarang介绍及使用
2018/02/07 Javascript
策略模式实现 Vue 动态表单验证的方法
2019/09/16 Javascript
vue滑动吸顶及锚点定位的示例代码
2020/05/10 Javascript
基于js实现的图片拖拽排序源码实例
2020/11/04 Javascript
Python操作mongodb数据库进行模糊查询操作示例
2018/06/09 Python
Django项目使用CircleCI的方法示例
2019/07/14 Python
Python 使用 Pillow 模块给图片添加文字水印的方法
2019/08/30 Python
简单了解python shutil模块原理及使用方法
2020/04/28 Python
python matplotlib库的基本使用
2020/09/23 Python
Python绘制K线图之可视化神器pyecharts的使用
2021/03/02 Python
利用CSS3的定位页面元素
2009/08/29 HTML / CSS
纯CSS3编写的的精美动画进度条(无flash/无图像/无脚本/附源码)
2013/01/07 HTML / CSS
关于HTML5的22个初级技巧(图文教程)
2012/06/21 HTML / CSS
HTML5 video 视频标签使用介绍
2014/02/03 HTML / CSS
教育专业自荐书范文
2013/12/17 职场文书
公积金转移接收函
2014/01/11 职场文书
2014党委书记四风对照检查材料思想汇报
2014/09/21 职场文书
2016年感恩教师节校园广播稿
2015/12/18 职场文书
2019暑假阅读倡议书
2019/06/24 职场文书
详细聊聊关于Mysql联合查询的那些事儿
2021/10/24 MySQL
SQL基础查询和LINQ集成化查询
2022/01/18 MySQL