如何理解 Python 中的面向对象编程?
创建Python类
数据属性
实例方法
属性
类和静态方法
继承
>>> class MyClass:
... pass
...
按照惯例,Python类的命名采用首字母大写(即PascalCase)。
>>> a = MyClass()
>>> a
<__main__.MyClass object at 0x7f32ef3deb70>
语句a = MyClass()创建了MyClass的一个实例,并将它的引用赋值给变量a。
>>> type(a)
<class '__main__.MyClass'>
>>> a.__class__
<class '__main__.MyClass'>
>>> a.__class__.__name__
'MyClass'
顺便提一句,Python类也是对象。它们是type的实例:
>>> type(MyClass)
<class 'type'>
下面,我们来定义一个方法。
>>> class MyClass:
... def __init__(self, arg_1, arg_2, arg_3):
... print(f'an instance of {type(self).__name__} created')
... print(f'arg_1: {arg_1}, arg_2: {arg_2}, arg_3: {arg_3}')
...
下面,我们来创建一个MyClass实例,看看这个初始化方法的具体工作。我们的.__init__()方法需要三个参数(arg_1、arg_2和arg_3),记住我们不需要传递与self对应的第一个参数。所以,在实例化对象时,我们需要传递三个参数:
>>> a = MyClass(2, 4, 8)
an instance of MyClass created
arg_1: 2, arg_2: 4, arg_3: 8
上述声明产生的结果如下:
创建一个MyClass类型的对象的实例。
自动调用该实例的方法.__init__()。
我们传递给MyClass()方法的参数:(2,4和8)会被传递给.__init__()。
.__init__()执行我们的请求,并输出结果。它利用type(self).__name__获取类的名称。
>>> class MyClass:
... def __init__(self, arg_1, arg_2, arg_3):
... self.x = arg_1
... self._y = arg_2
... self.__z = arg_3
...
现在MyClass有三个数据属性:
.x可以获取arg_1的值
._y可以获取arg_2的值
.__ z可以获取arg_3的值
>>> class MyClass:
... def __init__(self, arg_1, arg_2, arg_3):
... self.x, self._y, self.__z = arg_1, arg_2, arg_3
...
属性名称中的下划线(_)是为了表明这些属性是“私有”属性:
开头没有下划线的属性(比如.x)通常可供对象外部的调用和修改。
开头拥有一个下划线的属性(比如._y)通常也可以从对象外部调用和修改。然而,下划线是一种惯用的标志,即该类的创建者强烈建议不要使用该变量。应该仅通过类的功能成员(比如方法和属性)调用和修改该变量。
开头拥有双下划线的属性(比如.__ z)将在名字修饰过程中被改名(在本例中它将被改名为._MyClass__z)。你也可以通过这个新名称从对象外部调用和修改它们。但是,我强烈反对这种做法。应该尽通过类的功能成员以其原始名称进行调用和修改。
>>> a = MyClass(2, 4, 8)
>>> vars(a)
{'x': 2, '_y': 4, '_MyClass__z': 8}
>>> a.__dict__
{'x': 2, '_y': 4, '_MyClass__z': 8}
名字修饰过程把键'__z'变成了'_MyClass__z'。
>>> a.x
2
>>> a._y
4
>>> a.__z
Traceback (most recent call last):File "<stdin>", line 1, in <module>
AttributeError: 'MyClass' object has no attribute '__z'
>>> a.x = 16
>>> a.x
16
>>> vars(a)
{'x': 16, '_y': 4, '_MyClass__z': 8}
请注意,我们无法访问.__ z,因为.__ dict__没有键'__z'。
>>> class MyClass:
... def __init__(self, arg_1, arg_2, arg_3):
... self.x, self._y, self.__z = arg_1, arg_2, arg_3
...
... def set_z(self, value):
... self.__z = value
...
... def get_z(self):
... return self.__z
...
>>> b = MyClass(2, 4, 8)
方法.get_z()和.set_z()提供了传统的检索和修改.__ z值的方法:
>>> b.get_z()
8
>>> b.set_z(16)
>>> vars(b)
{'x': 2, '_y': 4, '_MyClass__z': 16}
你也可以在.get_z()和.set_z()中添加其他功能,例如检查数据的有效性。这种方法实现了面向对象编程中的一个主要概念:封装。
>>> class MyClass:
... def __init__(self, arg_1, arg_2, arg_3):
... self.x, self._y, self.__z = arg_1, arg_2, arg_3
...
... @property
... def z(self):
... return self.__z
...
... @z.setter
... def z(self, value):
... self.__z = value
...
>>> b = MyClass(2, 4, 8)
如下,我们利用相应的属性.z来访问和修改数据属性.__ z:
>>> b.z
8
>>> b.z = 16
>>> vars(b)
{'x': 2, '_y': 4, '_MyClass__z': 16}
这段代码比上述示例更精简优雅。
>>> class MyClass:
... def __init__(self, arg_1, arg_2, arg_3):
... self.x, self._y, self.__z = arg_1, arg_2, arg_3
...
... def f(self, arg):
... print('instance method f called')
... print(f'instance: {self}')
... print(f'instance attributes:\n{vars(self)}')
... print(f'class: {type(self)}')
... print(f'arg: {arg}')
...
... @classmethod
... def g(cls, arg):
... print('class method g called')
... print(f'cls: {cls}')
... print(f'arg: {arg}')
...
... @staticmethod
... def h(arg):
... print('static method h called')
... print(f'arg: {arg}')
...
>>> c = MyClass(2, 4, 8)
方法.f()是一个实例方法。实例方法的第一个参数是对象本身的引用。这些方法可以利用self访问对象,利用vars(self)或self.__dict__访问对象的数据属性,还可以利用type(self)或self.__class__访问对象对应的类,而且它们还可以拥有自己的参数。
>>> c.f('my-argument')
instance method f called
instance: <__main__.MyClass object at 0x7f32ef3def98>
instance attributes:
{'x': 2, '_y': 4, '_MyClass__z': 8}
class: <class '__main__.MyClass'>
arg: my-argument
通常,我们应该直接通过类(而不是实例)调用类方法和静态方法:
>>> MyClass.g('my-argument')
class method g called
cls: <class '__main__.MyClass'>
arg: my-argument
>>> MyClass.h('my-argument')
static method h called
arg: my-argument
请记住,我们不需要传递类方法的第一个参数:与cls相对应的参数。
>>> c.g('my-argument')
class method g called
cls: <class '__main__.MyClass'>
arg: my-argument
>>> c.h('my-argument')
static method h called
arg: my-argument
当我们调用c.g或c.h,但实例成员没有这样的名称时,Python会搜索类和静态成员。
>>> class MyOtherClass(MyClass):
... def __init__(self, u, v, w, x, y, z):
... super().__init__(x, y, z)
... self.__u, self.__v, self.__w = u, v, w
...
... def f_(self, arg):
... print('instance method f_ called')
... print(f'instance: {self}')
... print(f'instance attributes:\n{vars(self)}')
... print(f'class: {type(self)}')
... print(f'arg: {arg}')
...
>>> d = MyOtherClass(1, 2, 4, 8, 16, 32)
如上,MyOtherClass拥有MyClass的成员:.x、._y、.__z以及.f()。你可以通过语句super().__init__(x, y, z)初始化基类的数据成员x、._y和.__z,该语句会调用基类的.__init__()方法。
>>> vars(d)
{'x': 8,'_y': 16,'_MyClass__z': 32,'_MyOtherClass__u': 1,'_MyOtherClass__v': 2,'_MyOtherClass__w': 4}
我们可以调用基类和派生类中的所有方法:
>>> d.f('some-argument')
instance method f called
instance: <__main__.MyOtherClass object at 0x7f32ef3e7048>
instance attributes:
{'x': 8,'_y': 16,'_MyClass__z': 32,'_MyOtherClass__u': 1,'_MyOtherClass__v': 2,'_MyOtherClass__w': 4}
class: <class '__main__.MyOtherClass'>
arg: some-argument
>>> d.f_('some-argument')
instance method f_ called
instance: <__main__.MyOtherClass object at 0x7f32ef3e7048>
instance attributes:
{'x': 8,'_y': 16,'_MyClass__z': 32,'_MyOtherClass__u': 1,'_MyOtherClass__v': 2,'_MyOtherClass__w': 4}
class: <class '__main__.MyOtherClass'>
arg: some-argument
但是,如果派生类包含的某个成员与基类同名,则优先使用派生类的成员。
方法:.__repr__()和.__str__()
方法:.__new__()
操作符
方法:.__getattribute__()、.__getattr__()、.__setattr__()和.__delattr__()
生成器
可调用性
创建序列
描述器
上下文管理
抽象类和成员
多重继承
使用super()
拷贝
序列化
slot
类修饰器
数据类
【END】
如何在短时间内成为Python工程师?
https://edu.csdn.net/topic/python115?utm_source=csdn_bw
热 文 推 荐
如何理解 Python 中的面向对象编程?相关推荐
- python采用面向对象编程模式吗_如何理解 Python 中的面向对象编程?
现如今面向对象编程的使用非常广泛,本文我们就来探讨一下Python中的面向对象编程. 作者 | Radek Fabisiak 译者 | 弯月,责编 | 郭芮 以下为译文: Python支持多种类型的编 ...
- python如何初始化对象数组_如何理解Python中的面向对象编程?
(由Python大本营付费下载自视觉中国) 作者 | Radek Fabisiak 译者 | 弯月,责编 | 郭芮 出品 | CSDN(ID:CSDNnews) 现如今面向对象编程的使用非常广泛,本文 ...
- 如何理解Python中的面向对象
一.认识面向对象是什么 面向过程的程序设计的核心就是过程,就是流水线式的思维,过程就是解决问题的步骤,面向过程的设计就好像一条设计好的流水线,考虑周全什么就处理什么东西. 优点在于极大地降低了写程序的 ...
- 简述python中面向对象的概念_简述Python中的面向对象编程的概念
面向对象编程--Object Oriented Programming,简称OOP,是一种程序设计思想.OOP把对象作为程序的基本单元,一个对象包含了数据和操作数据的函数. 面向过程的程序设计把计算机 ...
- python中对象的概念是什么_简述Python中的面向对象编程的概念
面向对象编程--Object Oriented Programming,简称OOP,是一种程序设计思想.OOP把对象作为程序的基本单元,一个对象包含了数据和操作数据的函数. 面向过程的程序设计把计算机 ...
- python中面向对象_简述Python中的面向对象编程的概念
面向对象编程--Object Oriented Programming,简称OOP,是一种程序设计思想.OOP把对象作为程序的基本单元,一个对象包含了数据和操作数据的函数. 面向过程的程序设计把计算机 ...
- python中面向对象的ui_怎样理解Python中的面向对象?
面向对象(Object Oriented)是程序开发领域中的重要思想,这种思想模拟了人类认识客观世界的逻辑,是当前计算机软件工程学的主流方法:类是面向对象的实现手段.Python在设计之初就已经是一门 ...
- 深入理解Python中的面向对象
博客核心内容: 面向过程与面向对象的对比 id.type和value的讲解 类和对象的概念 初始化构造函数__init__的作用 self关键字的使用 继承的概念 组合的概念 接口的概念 抽象类的概念 ...
- Python中的面向对象编程(类编程)由简单到复杂的示例代码
关于本文代码中的self的含义,大家可参考下面这篇博文: https://blog.csdn.net/wenhao_ir/article/details/125384347 另:说明一下,本博文中&q ...
最新文章
- 2014 年美国程序员薪资调查
- 分布式技术追踪 2017年第十二期
- loadrunner中创建唯一随机数
- pyinstaller3.5 和 python 3.8 不兼容
- yum 安装服务出现报错收集
- matlab mbuild setup,关于mbuild的一个问题
- 神策与CDA,一次不平凡的约会
- 超全药理学问答题汇总
- node进程管理——pm2
- JavaScript中一个对象数组按照另一个数组排序
- java tlv格式_java解析TLV格式数据
- 【error】error: failed to push some refs to ‘远程仓库地址‘ git报错解决
- iPhone12或再引领轻薄时尚风潮,半斤机将被抛弃
- 计算机组成原理-宝典
- 漫画:卖鱼与买鱼之生产与消费
- Godot Engine:将Sprite和刚体保持同一轴心
- 【评测】一种组织蛋白快速提取方法
- vue 实现上拉加载下拉刷新(思路贼清晰)
- CUDA计算能力显卡对照表
- AMA预告|章鱼加速器如何在熊市助力 Web3 创业
热门文章
- 当你输入一个网址的时候,实际会发生什么?(转)
- Ruby eventmachine install
- Access denied for user 'root'@'192.168.64.154' (using password: YES)
- [论文阅读] Instance-level salient object segmentation
- matlab线圈磁场,利用Matlab计算螺线管内磁场分布研究.pdf
- access mysql连接字符串_[数据库连接字符串] Access 连接字符串
- 【C++笔记】表达式 语句
- 搭建、使用与维护私有PyPi仓库
- Flutter进阶—质感设计之列表项
- 2021年中国住宅按揭服务市场趋势报告、技术动态创新及2027年市场预测