查看原文
其他

Python类中的方法是如何工作的

2018-01-14 刘志军 Python爱好者社区

作者:刘志军,6年+Python使用经验, 高级开发工程师,目前在互联网医疗行业从事Web系统构架工作

个人公众号:Python之禅(微信ID:vttalk)

题图:https://unsplash.com/@nielsenramon

在OO(面向对象)编程中,类中的方法有多种形式:实例方法、静态方法、类方法、甚至还可以有抽象方法,本文来说说实例方法在Python中是如何工作的,后面再来谈其他方法。

先来定义一个最简单类:

class Person:

   def __init__(self, name):
       self.name = name

   def eat(self):
       print(self) # <__main__.Person object at 0x00  
       print(type(self))  # <class '__main__.Person'>
       print(self.name + " is eating")

这里的 eat 就是一个实例方法,跟普通函数差不多,唯一的不同是必须指定一个参数 self,尽管名字可以任意命名,但约定俗成的叫 self,self 是什么?它代表Person类的实例对象,就像Java中的this一样,看下面的测试代码

p = Person("zhangsan")
p.eat()

p与self指向同一个实例对象

那么可不可以通过类直接调用呢?不行!

Person.eat() TypeError: eat() missing 1 required positional argument: 'self'

那为什么通过实例p调用eat方法不需要传递self参数呢?这个就要从函数与方法的区别说起。来看看下面的代码:

print(Person.eat)
print(p.eat)

# 输出
<function Person.eat at 0x000001BB242AAAE8>
<bound method Person.eat of <__main__.Person object at 0x000001BB242B4B38>>

前者是函数,后者是方法,有人说函数定义在类外面,方法定义在类里面,显示这种说法不全面,那么他们的区别在哪里?

首先方法是与某个对象相关联的,而函数则不是,p.eat 就是一个绑定了实例对象的方法,函数的所有参数都需要显示地传递,而方法中的数据是隐式传递的。Person.eat是函数,参数要显示地传递,Person.eat(p)

而方法因为绑定了实例对象,所以他调用的时候无需再传递实例对象了,直接调用p.eat()就可以了,self参数Python会自动传递过去,如果重复传递会报错。

p.eat(p)

TypeError: eat() takes 1 positional argument but 2 were given

所以,本质上

p.eat() 等价于 Person.eat(p)

那么对于实例方法,self 参数从语言设计的角度来说,是不是可以去掉呢,这个问题 Python 之父 Guido van Rossum 撰文解释过这件事,理由是 “Explicit is better than implicit”

延伸阅读:http://neopythonic.blogspot.de/2008/10/why-explicit-self-has-to-stay.htmlPython爱好者社区历史文章大合集

Python爱好者社区历史文章列表(每周append更新一次)

福利:文末扫码立刻关注公众号,“Python爱好者社区”,开始学习Python课程:

关注后在公众号内回复“课程”即可获取:

小编的Python入门视频课程!!!

崔老师爬虫实战案例免费学习视频。

丘老师数据科学入门指导免费学习视频。

陈老师数据分析报告制作免费学习视频。

玩转大数据分析!Spark2.X+Python 精华实战课程免费学习视频。

丘老师Python网络爬虫实战免费学习视频。


您可能也对以下帖子感兴趣

文章有问题?点此查看未经处理的缓存