京公网安备 11010802034615号
经营许可证编号:京B2-20210330
浅谈python中的面向对象和类的基本语法
当我发现要写python的面向对象的时候,我是踌躇满面,坐立不安呀。我一直在想:这个坑应该怎么爬?因为python中关于面向对象的内容很多,如果要讲透,最好是用面向对象的思想重新学一遍前面的内容。这个坑是如此之大,犹豫再三,还是只捡一下重要的内容来讲吧,不足的内容只能靠大家自己去补充了。
惯例声明一下,我使用的版本是 python2.7,版本之间可能存在差异。
好,在开讲之前,我们先思考一个问题,看代码:
为什么我只创建是为 a 赋值,就可以使用一些我没写过的方法?
可能会有小伙伴说:因为 a 此时是个字符串对象呀,当然能够使用字符串的方法,至于这些方法,那是python事先写好的。
好吧,那换个问题,为什么 python 知道它是个字符串对象?
在深入讲这个问题之前,先记住一句话:python中一切皆对象,对象都是由类创建的。
那么类是什么东西呢?我在这举个例子。
有一天,我在街上走着,突然看见前面有一个仇人。此时我想揍他一顿,但是力是相互作用的呀,打他我的拳头也会疼。诶,此时我发现路边有个石头,大小形状正合适,在不考虑警察叔叔怎么想的前提下,这是一个好的选择。然后我抄起石头,对着仇人进行了一些好孩子不能模仿的行为,然后深藏功与名,收工回家。
在这个过程中,我们为什么会知道那块东西是石头?
因为它的具有石头的外观特征呀!
为什么我们会知道石头可以砸人?
因为石头硬呀!
为什么我们知道石头是硬的?
因为......嗯.....小时候我爸告诉我的。
此时可以得到一个结论:我们知道一个东西是什么,具有什么功能,是因为我们的脑海中已经有个这个东西的概念。而这个概念可能是从长辈那里知道的,也可能是自己各种实验后自己总结的。类似于我们对于‘黑洞'这个东西的认识,还是科学家的各种研究总结而来。
如何将用石头打人这个例子用代码来实现的话:
class Stone(object): # 我创建一个叫Stone的类
def attack(self):
print '把头伸过来,我给你加个buff'
a = Stone() # 我用类创建了一个对象,也称为类的实例化
a.attack() # 我使用这个对象的方法

很好,这样我们就完成了我们的需要了。
如果我们再创建其他对象:

很显然python并没有为我们准备 attack 方法,所以就用不了。
所谓的类只是一个抽象的定义,而实例则是具体的对象。它们之间的差别就想我脑海中的石头和我手中的石头一样,只有后者才真实存在。当然,这里不要跟我讲什么唯心主义论什么的。
看到这里,关于类和对象之间的关系应该清楚了一些。有些同学可能会问:python中的类都是我们事先写好的,有没有方法让python自动生成类,然后在特定的时候使用这些类呢?恭喜你,同学,你可能摸到了人工智能的门槛了,如果能够实现的话,那和我们人类的学习能力不是差不多吗?好吧,这只是我的一个设想,真正的人工智能要怎么实现,我也不知道,还没有这么高的水平去研究先,不过作为一名程序员,能够编写人工智能,可能是终身的追求了吧。
在讲完类和实例化是怎么一回事之后,我们来看看类的基本语法。
首先,和定义函数的 def 一样, class 是定义类的关键字。
紧接着的是类名,这个可以自定义,同样的,不能和python的内置关键字冲突。另外,建议避开python的内建类型,例如 str、int之类的名字。规范的命名应该遵从“驼峰命名法”,例如: MyClass 这里的命名,每个单词的首字母大写。
然后是一个括号,里面的参数是用于继承的,一般继承于 object,表示一个新式类。另外,你可能见过没有括号的写法,这是经典类的写法。
示例:
class NewClass(object):
pass
class OldClass:
pass
New = NewClass() # 创建一个新式类的实例
Old = OldClass() # 创建一个经典类的实例
这就是类的基本语法,当然这样还是不够的,但是在更深入之前,我想先讲一个新旧式类的差别。
在这里,我们先打印一下两个变量的类型:
print type(New)
print type(Old)

可以看下两者的输出是不同的。
在早于python2.2的版本时,只有经典类这一种写法,当时,类和类型没有合并。
类是类对象,实例是实例对象,这两个对象之间没有任何关系。
这句话是什么意思?看代码:
print type(OldClass)
print type(Old)

我们可以看见其输出很含糊,经典类属于类对象,无论是哪个类,都统一为“类”类型,实例属于实例类型,却不知道其是由哪个类创建的,所以的实例都统一为“实例”类型。也就是说当时的类型用 classobj 和 instance 代表了所以的类和实例,无论你是哪个类,又或是哪个类创建的实例。
这样的信息实在太少,而类和类型之间非常混乱。为了解决这种情况,在 python2.2 中引入了新式类,并进行了类和类型的同统一。
print type(NewClass)
print type(New)

类的类型是 type?type 返回的对象还能像类一样创新新对象?
总结的来说:在新式类中,所以的类对象都是 type 的实例。而不同的类对象有能创建出其对应的实例。
class NewClass(object):
def __init__(self, val):
self.val = val
New = NewClass(123)
b = type(New)(321) # 对实例来说type返回的是类对象,我又可以用类对象来和创建新的实例
print b.val

构造器方法
一般可以理解类中的函数就是方法,而方法分为:实例方法,只有实例化后才能调用的,其第一个参数一般为 self,代表实例本身;类方法,其第一个参数为 cls,代表类本身;还有静态方法,就是个普通函数,没有要求参数。
1. __init__(self [,arg1,....]):
当类被调用进行实例化的时候,python会自动调用类里面的构造函数(如果有的话),在构造函数中,可以进行各种初始化的操作,最常见的就是上面的进行实例的属性的创建。
python 在示例化的时候,会检查其实行了 __init__ 方法了没有,如果没有则不对实例进行任何操作,然后返回对象。如果实行了这个方法,则自动调用这个方法,并自动将 self 传进行,也就是说我们在实例化进行传参的时候,将不用理会 self,直接传给后面的参数。
讲到属性,就必须要提一下什么是属性。属性这个对象其实更像一个变量,大多数对象都可以有属性(不包括python的内置类型),例如函数。
def Test():
pass
Test.a = 123
print Test.a

因为函数也是一个对象。
属性在类中,就是一个变量,例如:
class NewClass(object):
a = 123
print NewClass.a

当然,因为 python 的特性,我们可以在运作中为某个对象添加属性,而不用一开始就在类中写定。
注意,这个方法应该返回 None,也就是说我们一般不用 return 任何对象,让它默认返回就行了。
2. __new__(cls [,arg1,....]):
这也是一个构造器方法,它是一个类方法,一般在对 python 的不可变数据类型进行继承扩展的时候用的比较多。
某处拿来的代码示例:
class RoundFloat(float):
def __new__(cls, val):
return super(RoundFloat, cls).__new__(cls, round(val, 2))
a = RoundFloat(3.14159)
print a

解构器方法
__del__(self [,arg1,....])
这个方法将会在对象所以的引用被清除后才执行,例如:
class Test(object):
def __del__(self):
print '我被干掉了,兄弟们为我报仇!'
a = Test() # 创建了一个对象
b = a # b又引用了a
c = b # c又引用了b,现在 a 所指向的对象有3次引用,相当有三条命
del a # 干掉一条命
del b # 又干掉
del c # 听说你有3条命?全部干掉!

注意,这里只输出了一次,也就是说到了最后才删除完毕。这里要注意一下几点:
1.调用 del 并不意味着完成删除某个对象,只是减少引用。
2.如果你有一个循环引用或其它的原因,让一个实例的引用逗留不去,该对象的__del__()可能永远不会被执行。
3.__del__()未捕获的异常会被忽略掉(因为一些在__del__()用到的变量或许已经被删除了)。 不要在__del__()中干与实例没任何关系的事情。
4.一般情况下并不用实现这个方法,因为这样有一定的风险。
5.如果你定义了__del__,并且实例是某个循环的一部分,垃圾回收器将不会终止这个循环— —你需要自已显式调用 del。
6.如果继承了父类,且父类中也有解构器,要记得调用。否则可能会有某些在父类中的清理方法没有调用到,出现以下无法预料的错误。
数据分析咨询请扫描二维码
若不方便扫码,搜微信号:CDAshujufenxi
在数据驱动决策的时代,数据分析已成为企业运营、产品优化、业务增长的核心工具。但实际工作中,很多数据分析项目看似流程完整, ...
2026-01-15在CDA(Certified Data Analyst)数据分析师的日常工作中,“高维数据处理”是高频痛点——比如用户画像包含“浏览次数、停留时 ...
2026-01-15在教育测量与评价领域,百分制考试成绩的分布规律是评估教学效果、优化命题设计的核心依据,而正态分布则是其中最具代表性的分布 ...
2026-01-15在用户从“接触产品”到“完成核心目标”的全链路中,流失是必然存在的——电商用户可能“浏览商品却未下单”,APP新用户可能“ ...
2026-01-14在产品增长的核心指标体系中,次日留存率是当之无愧的“入门级关键指标”——它直接反映用户对产品的首次体验反馈,是判断产品是 ...
2026-01-14在CDA(Certified Data Analyst)数据分析师的业务实操中,“分类预测”是高频核心需求——比如“预测用户是否会购买商品”“判 ...
2026-01-14在数字化时代,用户的每一次操作——无论是电商平台的“浏览-加购-下单”、APP的“登录-点击-留存”,还是金融产品的“注册-实名 ...
2026-01-13在数据驱动决策的时代,“数据质量决定分析价值”已成为行业共识。数据库、日志系统、第三方平台等渠道采集的原始数据,往往存在 ...
2026-01-13在CDA(Certified Data Analyst)数据分析师的核心能力体系中,“通过数据建立模型、实现预测与归因”是进阶关键——比如“预测 ...
2026-01-13在企业数字化转型过程中,业务模型与数据模型是两大核心支撑体系:业务模型承载“业务应该如何运转”的逻辑,数据模型解决“数据 ...
2026-01-12当前手游市场进入存量竞争时代,“拉新难、留存更难”成为行业普遍痛点。对于手游产品而言,用户留存率不仅直接决定产品的生命周 ...
2026-01-12在CDA(Certified Data Analyst)数据分析师的日常工作中,“挖掘变量间的关联关系”是高频核心需求——比如判断“用户停留时长 ...
2026-01-12在存量竞争时代,用户流失率直接影响企业的营收与市场竞争力。无论是电商、互联网服务还是金融行业,提前精准预测潜在流失用户, ...
2026-01-09在量化投资领域,多因子选股是主流的选股策略之一——其核心逻辑是通过挖掘影响股票未来收益的各类因子(如估值、成长、盈利、流 ...
2026-01-09在CDA(Certified Data Analyst)数据分析师的工作场景中,分类型变量的关联分析是高频需求——例如“用户性别与商品偏好是否相 ...
2026-01-09数据库中的历史数据,是企业运营过程中沉淀的核心资产——包含用户行为轨迹、业务交易记录、产品迭代日志、市场活动效果等多维度 ...
2026-01-08在电商行业竞争日趋激烈的当下,数据已成为驱动业务增长的核心引擎。电商公司的数据分析师,不仅是数据的“解读官”,更是业务的 ...
2026-01-08在数据驱动决策的链路中,统计制图是CDA(Certified Data Analyst)数据分析师将抽象数据转化为直观洞察的关键载体。不同于普通 ...
2026-01-08在主成分分析(PCA)的学习与实践中,“主成分载荷矩阵”和“成分矩阵”是两个高频出现但极易混淆的核心概念。两者均是主成分分 ...
2026-01-07在教学管理、学生成绩分析场景中,成绩分布图是直观呈现成绩分布规律的核心工具——通过图表能快速看出成绩集中区间、高分/低分 ...
2026-01-07