python中function函数的用法_Python中的函数和方法,function,与,method

在编程语言中有两个很基础的概念,即方法(method)和函数(function)。

除去入参、返回值、匿名函数之类的正确的形式内容之外,

我们一般都会这样说:“函数就是定义在类外面的,而方法就是定义在类里面的,跟类绑定的”

深究一下

判断是函数(function)还是方法(method)

在标准库inspect 中,它提供了两个自省的函数,即 ismethod() 和 isfunction(),可以用来判断什么是方法,什么是函数。

import inspect

def test1():

print('这是方法还是函数?')

print(inspect.isfunction(test1)) # True

print(inspect.ismethod(test1)) # False

运行的结果分别是“True”和“False”,表明我们所定义的 test() 是一个函数,而不是一个方法。

注意:不能用这种方式,这种方式会调用函数

import inspect

def test1():

print('这是方法还是函数?')

print(inspect.isfunction(test1()))

# 输出结果

这是方法还是函数?

False

inspect.ismethod() 与 inspect.isfunction()

这两个函数也可以用来检测自身,不难验证出它们都是一种函数

import inspect

print(inspect.isfunction(inspect.ismethod)) # True

print(inspect.isfunction(inspect.isfunction)) # True

那么,inspect 库的两个函数是什么工作原理呢?

看看源码,将鼠标光标处放在要查看的内容上,按住

Ctrl

并单击鼠标左键,进入源码查看

通过 isfunction() 和 ismethod() 的注释,我们可以得到以下结论:

非用户定义的函数,即内置函数,在 isfunction() 眼里并不是“函数”(FunctionType)!

下面验证一下 len()、 range():

import inspect

print(inspect.isfunction(len)) # False

print(inspect.isfunction(range)) # False

可以发现,它们的本质并不是函数,事实上,它们有专属的类别(BuiltinFunctionType、BuiltinMethodType)

print(type(len))

输出结果:

内置函数都是builtin_function_or_method 类型

一个类的静态方法,在 ismethod() 眼里并不是方法(MethodType)!

class MyTest():

@classmethod

def cls_func(cls):

pass

def ins_func(self):

pass

@staticmethod

def sta_func():

pass

print(inspect.ismethod(MyTest.cls_func)) # True

print(inspect.ismethod(MyTest.ins_func)) # False

print(inspect.ismethod(MyTest.sta_func)) # False

创建了类的实例后,再看看

test = MyTest()

print(inspect.ismethod(test.cls_func)) # True

print(inspect.ismethod(test.ins_func)) # True

print(inspect.ismethod(test.sta_func)) # False

可以看出,除了 classmethod 之外,只有类实例的实例方法,才会被 ismethod() 判定为真!而静态方法,不管绑定在类还是实例上,都不算是“方法”!

在源码中,出现了 isinstance() 函数,它主要用于判断一个对象(object)是否是某个类(class)的实例(instance)。

还出现了 types.FunctionType 及types.MethodType ,它们指的就是目标类。继续点进去看源码

经过简化处理后,最关键是在于:通过type() 函数判断出一个对象是 function 或 method 类,

instance() 函数判断出一个对象是否为某个类的实例

所以

isfunction() 判断出的是用户定义的函数(user-defined function),

它拥有__doc__、__name__ 等等属性

ismethod() 判断出的是实例方法(instance method),

它拥有函数的一些属性,最特别的是还有一个 __self__ 属性

若以 inspect 库的两个函数为判断依据,则 Python 中的“方法与函数”具有一定的狭义性。

在判断什么是函数时,它们并不把内置函数计算在内。同时,在判断什么是方法时,并非定义在类内部的都算,而是只有类方法及绑定了实例的实例方法才算是“方法”。

也许你会说,inspect 的两个判断函数并不足信,规矩是写死的,也有不灵活的时候,内置函数也应该算是“函数”,类里面的所有方法都应该算是“方法”。

这种说法在广义上是可接受的,毕竟我们一直叫的就是“XX函数”、而不会去说“XX方法”。

但是,理论和广义概念只是方便人们的沟通理解,而代码实现才是本质的区别。

也就是说,Python 在实际区别“方法与函数”时,并不是文中开头所说:“函数就是定义在类外面的,而方法就是定义在类里面的,跟类绑定的”,还有更多的细节值得关注。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值