技术标签: 面向对象 OOP python 系统学习Python python面向对象 类
分类目录:《系统学习Python》总目录
从最底层来看,类基本上就是命名空间,这点很像Python的模块。但是类和模块不同的是,类还支持生成多个对象、命名空间继承以及运算符重载。
要了解多个对象的概念是如何工作的,我们需要先了解Python的OOP模型中的两种对象:类对象和实例对象:
各自都有独立的命名空间,但是同时继承了创建该实例的类中的变量名。类对象来自语句,而实例来自调用。每次调用一个类,你就会得到这个类的一个新的实例。
这种对象生成的概念和我们目前为止所见的大多数其他程序构件有着很大的不同。实际上,类是产生多个实例的工厂。相比之下,所有模块在程序中只能导入一个副本。事实上,这也就是我们需要用到reload
,它在原位置更新一个单例共享的模块对象。借助类,每个实例都有了它们自己的、独立的数据,从而支持类所建模型的对象的多个版本。
在这一角色下,类实例类似于闭包函数的基于调用的状态,但是这是类模型本身的一部分,并且类中的状态是显式的属性而不是隐式的作用域引用。此外,这只是类所做的部分工作——它们也支持通过继承实现的定制化、运算符重载,以及通过方法实现的多态行为。一般来讲,类是更加完整的编程工具,尽管OOP和函数式编程并不是相互排压的编程范式。我们通过在方法中使用函数化的工具,编写自身就是生成器的方法,以及编写用户定义的选代器等来将二者结合起来。
下面就是从类对象和实例对象出发,对Python中OOP本质的一个快速总结。正如我们将看到的,从某种程度上来说Python的类与def
和模块很相似,但是Python中的类和其他编程语言中的相比可能就大不相同了。
当我们执行class
语句时,就会得到类对象。以下是Python类的主要特性:
class
语句创建类对象并将其赋值给一个名称:就像函数的def
语句,Python的class
语句也是可执行语句。当其执行时会产生新的类对象,并被赋值给class
头部的名称。此外,就像def
一样,class
语句一般是在其所在文件被导入时执行的。class
语句内的赋值语句会创建类的属性:就像模块文件一样,class
语句内顶层的赋值语句(不是在嵌套的def
之内)会产生类对象中的属性。从技术角度来讲,class
语句定义了一个局部作用域,该作用域会变成类对象的属性的命名空间,就像模块的全局作用域一样。在执行class
语句后,类的属性可用点号加名称访问:object.name
。def
语句会生成方法,方法可用于处理实例。当调用类对象时,我们得到了实例对象。以下是类的实例的重点概要:
self
属性做赋值运算会产生每个实例自己的属性:在类的方法函数内,第一位参数(按惯例称为self
)会引用当前处理的实例对象。对self
的属性做赋值运算,会创建或修改实例内的数据,而不是类的数据。最终的结果就是,类定义了公用的、共享的数据和行为,并生成实例。实例反映了具体的应用程序中的实体,并记录了每个实例自己的随着对象变化的数据。
我们下面用一个真实的例子,来看看这些概念在实际中是如何工作的。首先,让我们编写一个Python的class
语句,来定义一个名为FirstClass
的类:
class FirstClass:
def setdata(self, value):
self.data = value
def display(self):
print(self.data)
一般来说这种语句是当其所在的模块文件导入时运行的。就像通过def
建立的函数,这个类在Python抵达并执行语句前是不会存在的。
就像所有复合语句一样,class
开头一行会列出类的名称,后面再接一个或多个内嵌并且缩进的语句的主体。在这里嵌套的语句是def
,它们定义了类要暴露出的行为的函数,def
本质上是赋值运算。在这里是在class
语句的作用域内,把函数对象赋值给名称setdata
和display
,从而产生附加在类上的FirstClass.setdata
和FirstClass.display
属性。事实上,所有在类嵌套代码块的顶层被赋值的名称,都会成为类的属性。
位于类中的函数通常称为方法。方法采用一般的def
语句编写,并且支持之前我们讲过的所有关于函数的内容,不过在方法函数被调用时,第一位参数自动接收隐含的实例对象,即调用的主体。我们需要创建一些类的实例来理解其工作原理:
x = FirstClass()
y = FirstClass()
以此方式调用类时会产生实例对象,而这些实例对象其实是可访问类属性的命名空间。可以说,我们此时有三个对象:两个实例对象和一个类对象。其实是有三个相链接的命名空间,如下图所示:
以OOP观点来看,我们说x
和y
都是FirstClass
对象,也就是说它们都继承了附加于类上的名称。这两个实例一开始是空的,但是它们被连接到创建它们的类。如果对实例以及类对象内的属性名称做点号运算,Python就会通过继承搜索访问类中的名称(除非该名称也位于实例内):
x.setdata('hy592070616')
y.setdata('MachineLearning')
×
或y
本身都没有setdata
属性,为了寻找这个属性,Python会顺着实例到类的连接搜索。而这就是所谓的Python的继承:继承是在属性点号运算时发生的,而且只与查找连接对象内的名称有关。在这里,是通过遵循上图的is-a
连接。
在FirstClass
的setdata
函数中,传入的值会被赋给self.data
。在方法中,self
会自动引用当前处理的实例(x
或y
),所以赋值语句会把值储存在实例的命名空间,而不是类的命名空间:这也就是上图中变量名data
的创建的方式。因为类可以产生多个实例,所以方法必须通过self
参数才能获取当前处理的实例。当调用类的display
方法来打印self.data
时,我们会发现每个实例的值都不相同。另外,名称display
在×
和y
中都相同,因为它是来自(或继承自)类的:
x.display()
y.display()
输出:
hy592070616
MachineLearning
注意,我们在每个实例内的data
成员储存了不同类型的对象,这里分别是字符串和浮点数。就像Python中的其他事物,实例属性(有时称作成员)并没有声明。与简单变量一样,实例在首次赋值后就会存在。事实上,如果在调用setdata
之前就对某一实例调用display
,则会触发名称未定义的错误,这是因为data
属性在setdata
方法赋值前,是不会在内存中存在的。
另一种理解该模型的动态性的方式是,我们可以在类的内部或外部修改实例属性。在类的内部,通过方法对self
进行赋值运算,而在类的外部,则可以通过对实例对象进行赋值运算:
x.data = 'hy592070616: MachineLearning'
x.display()
输出:
hy592070616: MachineLearning
虽然比较少见,但是我们甚至还可以通过在类方法函数外对变量名进行赋值运算,从而在实例命名空间内产生全新的属性:
x.blog = 'CSDN'
这样会增加一个名为blog
的新属性,实例对象x
的所有类方法都可以选择使用它。类通常可以选择通过对self
参数进行赋值运算从而建立实例的所有属性的。程序可以访问、修改或创建其所引用的所有对象的属性。
通常,添加类所不能使用的数据是没有意义的,因此我们可以基于属性访问运算符重载,使用额外的“私有”代码来阻止这种操作。不过,自由的属性访问使用了更少的语法,甚至有些情况下它是有用的——例如,在编写我们稍后看到的数据记录代码时。
文章浏览阅读299次。临邛道士鸿都客,能以精诚致魂魄。为感君王辗转思,遂教方士殷勤觅。排空驭气奔如电,升天入地求之遍。上穷碧落下黄泉,两处茫茫皆不见。忽闻海上有仙山,山在虚无缥缈间。楼阁玲珑五云起,其中绰约多仙子。中有一人字太真,雪肤花貌参差是。金阙西厢叩玉扃,转教小玉报双成。闻道汉家天子使,九华帐里梦魂惊。揽衣推枕起徘徊,珠箔银屏迤逦开。云鬓半偏新睡觉,花冠不整下堂来。1 类中的方法1.1..._python某市新建成了地铁,车票价格如下:1—4站2元,5—7站3元,8—9站4元,10站以上5
文章浏览阅读2.1w次,点赞7次,收藏8次。1,打开gedit,选择菜单栏->编辑(E)->首选项(E)2,进入编辑界面,点击查看按钮,设置显示行号等3,点击编辑器,设置制表符等4,点击字体和颜色,设置字体和颜色5,设置插件6.ok_文本编辑器getedit
文章浏览阅读4.6k次。package.json 文件是用来管理本地安装 npm 包的唯一文件。它描述了当前项目依赖的包的列表,包含项目依赖包的版本,便于项目管理和移植一、创建1、npm init 即可在当前目录创建一个 package.json 文件,跟着步骤选项,手动创建package.json文件2、npm init --yes 可以跳过回答问题步骤,直接生成默认值的 package.json 文件二..._vue package.json中的main
文章浏览阅读464次。缘起本期继续讨论边缘领域里的另一个有意思的东西——语言。正如前文所说,边缘设备是在终端、云端(或者叫后台系统)中出现的一种新形态的设备。它肯定不是终端,因为它比终端能力强,而且倾向于它是为..._边缘计算需要学什么语言
文章浏览阅读2.6k次,点赞4次,收藏7次。Spring项目中:类路径和Web应用的根路径最近在学习Spring书籍中提到:类路径和Web应用的根路径;百度一下,网上并没有给出直接的答案.最后根据百度的结果和自己的理解得出如下直接的答案.类路径:即发布到Tomcat服务器后的ClassPath路径:在IDEA中即resource路径(在Idea中有两个resource目录).Web应用的根路径:顾名思义,即是发布到Tomcat服..._springboot 项目根路径和类路径区别
文章浏览阅读1.3w次。介绍:有一个简单的需求,一个表格点击一个单元格的按钮后,按钮变成进度条。问题:如果只是用一个变量,然后配合v-if使用,一整列都会变成进度条。想法:把那个变量渲染进去列表的每一行。但是…问题又出现了,值变了,但是表格没有被重新渲染。重点:this.$set(this.tableData, index, row),更新表格这一行的值。理解这句代码的意思,后面的可以不看。随便放点前端代码就很长了,不放又清除概况,简单看看吧。直接看el-progress那个标签的附近的,其他的瞅一下知道大概就好。<_elementui刷新table
文章浏览阅读1.9w次,点赞2次,收藏3次。把Maven中央仓库换成阿里云后,下载速度快了很多!配置如下:修改maven根目录下的conf文件夹中的setting.xml文件,内容如下:< !-- 在mirrors 中配置 mirror --><mirrors> <mirror> <id>alimaven</id> <name>aliyun maven</name> <url>
文章浏览阅读2.1k次。最小二乘法拟合直线步骤:(1)分离出目标(如下二值图像中的黑色区域);(2)提取目标边缘坐标;(3)查找所要拟合的边缘坐标;(4)最小二乘法拟合直线。在本文中使用Zernike矩检测目标的亚边缘坐标。根据MATLAB中坐标数据的特点分离出所需坐标数据,这里以水平边缘坐标的纵坐标离散程度不大为查找的条件提取出水平边缘坐标。实验图和结果图如下所示:MATLAB实现:BW=imread("xx.jpg");Coor=zernike7(BW);%查找水平边缘的坐标delta=2;%控制坐标范_matlab拟合矩形
文章浏览阅读971次。高级数据结构实验(Kruskal算法)C语言实现实验目的:1.掌握并查集的合并优化和查询优化2.掌握Kruskal算法3.能够针对实际问题,能够正确选择贪心策略4.能够针对选择的贪心策略,证明算法的正确性5.能够正确分析算法的时间复杂度和空间复杂度实验内容:采用Kruskal算法生成最小生成树,并采用并查集的合并优化和查询优化。实验思路:使用Kruskal算法解决稀疏图解决最小生成树问题效率较高,并使用并查集这种数据结构实现了isConnected()和unionElements()两种_kruskal算法代码c语言实现
文章浏览阅读1.1k次。这里写自定义目录标题webService接口调用的通用方法具体方法如下如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居左、居右SmartyPants创建一个自定义列表如何创建一个注脚注释也是必不可少的KaTeX数学公式新的甘特图功能,丰富你的文章UML 图表FLowchart流程图导出与导入导出导入webService接口调用的通用方法webService接口调用的通用方法,一定要注意使用的URL中‘/’ 一定不能多,也一定不能少,否则会出现问题具体方法如下/**调用webS_调用webservice @webparam
文章浏览阅读1w次,点赞2次,收藏18次。最近开发小程序的过程中遇到了这样一个小问题,在wxss文件中通过background设置背景图片,要实现背景图片上显示文字的效果,发现背景图片不显示,设置背景图片的代码如下:.test { width:100rpx; height:30rpx; background:url('../../img/image.png') no-repeat; background-..._微信小程序背景出不来
文章浏览阅读892次。在清晨的八点,走出小区的大门,热气腾腾的市场,毫无规矩的车辆,一脸蜡像的人群,空气中隐藏的愿望,从未被提起的那些故事,那即将失去的时间,即将开始的生活,这个世界变得越难越难理解,如同城里开始变得越来越不如郊区,道路越修越堵,人们却越来越喜欢买车,写字楼盖得越来越多,却还是很难找到工作,淘宝上的二手货竟然会比新货的价格还高,而5块钱的汽油继08年之后竟然真的再次出现了,坐火车基本上..._熊熊作业我去