重拾计算机已经一个多月了,渐渐地对面向对象这一基础之地基有了进一步认识,想想大一暑假囫囵吞枣式的学习,真是可怜又可笑。最近读侯捷的《左手程序右手诗》受益匪浅,我自己都忘了多久没读过自传了,现在又心怀惆怅,总舍不得放不下自己的那可怜的几百本书,总想回去闭户读书,扔掉这可恶的计算机专业,可究竟自己不是周作人,也就作罢。
面向对象编程: 将程序推理为一组对象而不是一组操
更广泛地说,对象是程序中结合数据和代码的实体。一个对象封装一个状态,然后提供方法来操作该状态并与之交互。
状态: 内部信息的数据集合被称为对象的状态。
行为:对象可以执行的操作集合,通常是显示或修改它的内部状态。
客户端:与一个类或该类的多个对象交互的代码
类:描述某种类型的对象的状态和行为的蓝图
要删除Macin或Windows计算机上的文件,找到该文件图片并单机它。有几个操作选项,可以将其拖到垃圾箱/回收站中,或者可以从菜单中选择“删除”命令。这个是命令行范式的逆格式,在GUI中,它是“名词动词”结构。不同的交互形式是面向对象编程的核心。
数据属性:对象内部的变量,构成其内部状态的一部分
构造函数:一种特殊方法,用于在创建对象时初始化对象的状态
构造函数头看起来像函数或方法头,但它必须有__init__的特殊名称,在单词init的两侧有两个下划线。编写构造函数时,要指定客户端创建类型对象时必须传递的参数,以及如何使用这些参数来初始化新创建的对象。构造函数还接收名为self的初始参数,它表示正在创建的对象。
方法:一个可以对该对象进行操作的对象内部函数
方法就像一个独立存在于对象内部的函数。客户端通过方法与其对象的数据交互,以执行与该对象相关的有用行为
对象参数:引用被调用方法的对象的参数。在Python代码中通常命名为self
将此参数命名为self是一种惯例,但不是必需的。self不是Python的特殊关键字,但大多程序员始终使用此名称。
访问器和赋值器:
访问器提供有关对象状态的信息而不进行状态修改的方法。
赋值器修改对象内部状态的方法
当Python程序打印一个对象或将其转换为一个字符串时,程序会调用对象内置的一个名为__str__特殊方法,此方法不接受self以外的任何参数,并返回表示对象状态的字符串
当在对象上使用str转换函数时,也会调用__str__方法
当两个具有相同状态的对象相比较时,结果出乎意料地为False。这是因为Python不了解如何比较用户自定义类地对象,它只比较对象的引用值。默认情况下对象的引用值与其他对象的不同,即使恰好有相同的状态。
==运算符比较的是两个变量是否引用同一个对象,而不是两个不同的对象是否具有相同的状态,这涉及引用的概念
为了改变==和!=逻辑运算符,以便它们基于对象状态进行比较,通常定义一个名为__eq__的特殊谓词用法
__eq__方法执行两个对象的比较,如果二者状态相同则返回True。如Date类的实例对象的d1 == d2,d1 == d3 和d2 == d3全部都为True,当他们有相同的month和day值。
封装:对对象外部(客户端)隐藏对象内部的实现细节
抽象化:专注于基本属性而非内在细节
专注于收音机的外部行为使我们能够轻松使用它,而忽略了对我们来说不重要的内部工作细节。这是计算机科学中的一个重要概念——抽象化
——————————————————————
封装可以保护对象的状态免受不必要或无效的修改。
一些编程语言支持一种成为私有访问的概念,即外部代码无法直接访问对象的属性。Python不直接支持此类功能。Python有某些约定名称的属性被认为是私有属性。
定义此类属性的一般语法是在其名称前面加一个下划线:
class ClassName:
def __inti__(self, parameters):
self._name = value
下划线对客户端开发而言是一个私有属性的提示。
使用两个前导下划线不是一个下划线来命名属性,Python会对其值进行进一步保护。
Python解释器将在内部动态地重命名属性使其具有像_Date__month这样地混乱名称,使得客户端几乎不可能意外访问得到。
Python支持成为属性(property)的特殊方法,该方法被用作访问器来返回属性的值。属性方法的一个特殊方面是调用它的语法与直接访问数据属性的语法相同。
属性方法:用于访问或修改对象的数据属性的方法,调用它的语法与直接访问数据属性的语法相同
属性方法将直接访问数据属性的简单客户端语法与调用方法的封装语义相结合。(当一种语法为这样的操作提供简洁明了的语法时,它有时被称为“语法糖”。)
通过定义属性方法可以在自己的类中提供相同类型的功能。除了前面有一个@property的修饰符之外,定义属性方法的语法与编写方法相同,如下:
class ClassNmae:
@property
def property_name(self):
code to return the property's value
以下代码显示了对Date对象的month和day属性提供访问的属性方法:属性方法与其他访问器方法不同,因为客户端可以执行它们而无需再属性方法的名称后面加括号。这样做的好处是客户端可以使用更清晰的语法,更像是直接访问的属性,同时确保访问权限是只读的。
class Date:
def __init__(self, month, day):
self._month = month
self._day = day
@property
def month(self):
return self._month
@property
def day(self):
return self._day
授予对Date对象的month和day属性的访问权限似乎很奇怪,但是定义属性方法实际上并不违反对象的封装。这些属性方法只是将属性的副本返回给客户端,以便客户端可以查看month或者day值,但不能更改它们。
使用属性方法封装Date状态的一个缺点是客户端代码不再容易将Date设置为新的月份或日期。这样的设计目的是阻止客户端将Date设置为有效范围之外的值。在大多数其他语言中,此问题解决方案是为每个属性编写一个赋值(mutator)方法,其名称类似于set_month,set_day,Python提供了一种更好的机制,称为属性赋值器,允许在设置和访问属性的值时使用属性方法的语法。
class Date:
@property
def month(self):
return self._month
@month.setter
def month(self, value):
# self._month = value
if value >= 1 and value <= 12:
self._month = value
else:
raise ValueError("Invalid month: " + str(value))
参考自《Python程序设计与算法思维》