关于Python函数与方法的区别

举报
技术火炬手 发表于 2019/08/12 10:07:55 2019/08/12
【摘要】 今天重点跟大家对比下实例方法、类方法、静态方法。

1.png

每当逢年过节收假后,大家都经常会出现假期焦虑症的问题,但我不一样,我有星期焦虑症,哪怕周末只放一天,想到第二天要上班都会莫名的焦虑。原因无它,通篇一个懒字直达主题

640.gif

2.png

很多童鞋会比较疑惑,在python中方法和函数有什么区别,都是通过def去定义的啊?其实不然,今天我们就来分别介绍下两者…

在学习之前先来看下面的一段代码:

class Yoo:

   def hi(self):
       print("Guess it...")

上面的代码,在正常运行的前提下,hi是方法还是函数,你们怎么看?

多数朋友会觉得,这还用问么,在类中定义的一定就是方法,函数是单独定义的啊。如果你有这种想法,那就太可怕了!

类型判断

如果我们想判断一个对象,是什么类型,初学者都会使用type,而方法和函数,在type中已有定义,分别为MethodType,FunctionType,按照上面所说,我们进行下判断

# hi为方法
x = Yoo()
isinstance(x.hi, MethodType) >>> True

# hi为函数
isinstance(Yoo.hi, FunctionType) >>> True

为什么同样的代码,不同的调用方式,会得到不同的结果呢?这还怎么区别方法与函数啊!其实很简单,大家只需要记住一句话:

方法,是用来描述一个对象的行为动作!

第一种方式是类的对象调用hi即为方法,第二种是类调用hi则为函数….

3.png

函数

关于python的函数,大体可分为内置函数、匿名函数、自定义函数….

内置函数: 听名字就知道python为我们默认设置的一些函数,比如type、range,等等…全量的可以看这里https://www.runoob.com/python/python-built-in-functions.html

匿名函数: python匿名函数lambda将简单的多行命令转化为单行执行,简直不要太方便。

自定义函数: 用户自己定义的函数….

方法

python的方法,就比较丰富了,你可以这么划分它:

实例方法 通过self进行调用的方法

属性方法 通过装饰器property来定义的getattr和setattr的属性方法

魔法方法 又称为特殊方法,特征明显,即通过双下划钱使用的方法

私有方法 在方法前添加两个下划线定义的方法

类方法 通过类名的调用去操作公共模板中的属性和方法

静态方法 不用传入类空间、对象的方法, 作用是保证代码的一致性,规范性,可以完全独立类外的一个方法

今天重点跟大家对比下实例方法、类方法、静态方法,让我们来看一段代码

class MainClass:
   def shilifangfa(self):
       print('这是一个实例方法')

   @classmethod
   def leifangfa(cls):
       print('这是一个类方法')
   @staticmethod
   def jingtaifangfa():
       print('这是一个静态方法')

m=MainClass()
m.shilifangfa()
m.leifangfa()
m.jingtaifangfa()

output:
这是一个实例方法
这是一个类方法
这是一个静态方法

好吧,先说下方法的分类吧:

  1. 实例方法

  2. 类方法

  3. 静态方法

5.png

可是,这三种方法有什么区别呢,光看上面的这段代码,没有任何差异啊。

那么我们把这段代码稍作修改

class MainClass:
   def shilifangfa(self):
       print('这是一个实例方法',self)

   @classmethod
   def leifangfa(cls):
       print('这是一个类方法',cls)
   @staticmethod
   def jingtaifangfa():
       print('这是一个静态方法')

m=MainClass()
m.shilifangfa()
m.leifangfa()
m.jingtaifangfa()
output:
这是一个实例方法 <__main__.MainClass object at 0x00000000012AF0B8>
这是一个类方法 <class '__main__.MainClass'>
这是一个静态方法

这次的输出和上次相比,是否看到实例方法,我们需要传给它一个实例对象,而类方法我们需要传给他的是一个类

那么如果我们给实例方法传输的不是一个由类产生的实例会如何呢?

image.png

报错的意思很明显,我们需要传给他一个实例的名称,当然你也可以随便写一个数值去传递,那么也就没有作为实例方法的意义了

说了实例方法,那么类方法呢?同样我们换一种方式去调用

image.png

MainClass.leifangfa()

这是一个类方法

6.png

这次没有任何变化,为什么呢?

原因在于,当使用类方法时,传递的参数应该是一个类名,但如果是实例,系统会自动找到实例对应的类名称

仔细看看下面的代码,你会对实例方法与类方法有更详细的认识

class MainClass:
   a=100 # 方法本身的属性
   def shilifangfa(self):
       print('这是一个实例方法',self.a)
       print(self.index) # 正常访问
       self.a=200 # 相当于实例本身创建了一个属性
       print(self.a) # output a=200
       print(MainClass.a) # output a=100


   @classmethod
   def leifangfa(cls):
       print(cls.index) # type object 'MainClass' has no attribute 'index'
       # 注意这句type object 它变相的告诉你,类也是一个对象,只不过是一个type的对象
       # 这就更验证了python中的那句,万物皆对象
       print('这是一个类方法',cls.a)
       cls.a = 200 # 修改了类本身的a
       print(cls.a)  # output a=200
       print(MainClass.a) # output a=200
       # print(cls.index) # 类方法无法获取到实例的属性
   @staticmethod
   def jingtaifangfa():
       print('这是一个静态方法')

m=MainClass()
m.index=1000 # 实例本身的属性
m.shilifangfa()
MainClass.leifangfa()
m.jingtaifangfa()

4.png

欢迎大家关注清风Python公众号

分割线动图.gif

本文来自“清风Python”公众号



【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。