百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

Python之函数高级:函数深度剖析,一次性搞懂函数对象

off999 2024-09-21 20:49 35 浏览 0 评论

引言

在之前的系列文章中,我们介绍过了函数的常见用法,以及Python中一切皆对象,函数也是对象的概念。在函数高级的前两篇文章中,我们回顾并补充论述了函数对象是一等公民,以及Python中的高阶函数的概念。

为了为后续对Python中闭包内容的介绍打基础,今天的文章中,我打算换个角度,深入剖析下函数对象的内部构成。

本文的主要内容有:

1、函数对象的主要属性

2、函数对象的主要方法

3、函数对象中的代码在哪里


函数对象的主要属性

函数在Python中是作为对象的存在的,对象一般会有相应的属性和方法,我们首先来看下Python中有哪些主要的属性,从而更加清晰地理解函数对象。

首先看下函数对象所属类的定义:

从定义中可以看出,Python中的函数对象有这么几个属性,简单列举说明一下:

1、__name__:字符串类型,存储的是函数对象的名称。

2、__qualname__:字符串类型,存储的是方法对应的函数对象的全限定名。

3、__module__:函数对象所在的模块的名称

3、__defaults__:以元组形式存储函数对象中位置参数的默认值。

4、__kwdefaults__:以字典形式存储函数对象中的强制关键字参数的默认值。

5、__globals__:property属性,存储函数对象所在的全局命名空间的名字绑定的字典。

6、__closure__:property属性,存储函数对象构建的闭包中的自由元素(自由变量、内部函数对象)。

7、__code__:存储函数编译后的字节码和其他相关信息。

接下来演示下相关函数对象的属性,直接看代码:

首先看入口文件中定义的函数对象的相关属性情况:

from rich.pretty import pprint as print

a = 20


def add(n=1000, *, m=23000):
    b = 10
    c = 20
    return a + n


print(add.__name__)
print(add.__qualname__)
print(add.__module__)
print(add.__defaults__)
print(add.__kwdefaults__)
print(add.__code__)
print(add.__closure__)
print(add.__globals__)
# 与globals()返回的全局命名空间中的名字对象绑定的字典对象是同一个对象
print(add.__globals__ is globals())

执行结果:

然后,我们定义一个名为m1.py的模块:

a = 8888


def add(a, b):
		return a + b


class Test:
    def hello(self):
        print(self)

然后,我们在入口文件导入该模块,查看m1模块中的add()函数的属性:

import m1

a = 100

print(m1.add.__name__)
print(m1.add.__qualname__)
print(m1.add.__module__)
print(m1.add.__defaults__)
print(m1.add.__kwdefaults__)
print(m1.add.__code__)
print(m1.add.__closure__)
# 与globals()返回的全局命名空间中的名字对象绑定的字典对象是同一个对象
print(m1.add.__globals__ is globals())
# __globals__中的内容有点多,只看下我们关心的内容
print({kk: vv for kk, vv in m1.add.__globals__.items() if not kk.startswith('_')})
# globals()中的内容,只看我们关心的内容
print({kk: vv for kk, vv in globals().items() if not kk.startswith('_')})

执行结果:

最后我们再看一下特殊的函数对象,即类中方法的相关属性:

import m1

a = 100

test = m1.Test()
print(test.hello.__name__)
print(test.hello.__qualname__)
print(test.hello.__module__)
print(test.hello.__defaults__)
print(test.hello.__kwdefaults__)
print(test.hello.__code__)
print(test.hello.__closure__)
# 与globals()返回的全局命名空间中的名字对象绑定的字典对象是同一个对象
print(test.hello.__globals__ is globals())
# __globals__中的内容有点多,只看下我们关心的内容
print({kk: vv for kk, vv in test.hello.__globals__.items() if not kk.startswith('_')})
# globals()中的内容,只看我们关心的内容
print({kk: vv for kk, vv in globals().items() if not kk.startswith('_')})

执行结果:


从入口文件中定义的函数、导入模块中的函数,以及类中的方法,这三种不同的函数对象的实际代码演示,我们再稍微总结一下函数对象中的属性相关内容:

1、__name__在三种函数对象中都是函数定义时的名称

2、__qualname__只有在类中定义的方法中,变为“类名.方法名”的形式,其他情况都与__name__属性相同。

3、__defaults__和__kwdefaults__的区别在于,前者存储位置参数的默认值,是一个元组形式,而后者存储的是函数定义中,在*之后定义的强制关键字形式传递的参数的默认值,是以字典形式存在的。

4、__globals__属性,是定义该函数对象的全局命名空间对应的字典对象,所以,入口文件中定义的函数的__globals__属性与入口文件中的globals()返回的字典是同一个对象。而模块中定义的函数的__global__属性存储的是模块对应的全局命名空间所对应的字典对象,与入口文件的globals()返回的字典对象不同。

5、__closure__属性,会用于存储闭包中的自由变量及内部函数对象,从而使得闭包得以建立,这个会在后面的闭包内容中进一步展开。

6、__code__属性,存储函数定义相关代码编译为的字节码对象,本文会稍后进一步展开。


函数对象的主要方法

看完了函数对象中的比较重要的属性,接下来,我们看下函数对象的两个方法:__init__()和__call__()方法。

首先看下__init__()方法的描述:

从函数对象的__init__()初始化方法的相关定义,我们是否能通过别的方式初始化一个函数对象呢?

接下来,我们尝试基于一个已经定义好的函数的相关属性,作为参数,重新初始化一个新的函数对象。

直接看代码:

a = 10
b = 20


def add(n1, n2):
		return a + b + n1 + n2


# 通过函数对象的__class__属性获取函数类,然后进行对象的实例化
# 也可以通过type(add)获取函数对象的类,进行对象的实例化,效果是一样的
# new_add = type(add)(code=add.__code__, globals={'a': 100, 'b': 200}, name='new_add')
new_add = add.__class__(code=add.__code__, globals={'a': 100, 'b': 200}, name='new_add')
print(add)
print(add.__name__)
print(add.__code__)
print(add(1, 2))
print(new_add)
print(new_add.__name__)
print(new_add.__code__)
print(new_add(1, 2))

执行结果:

从执行结果可以看出:

1、除了通过def关键字实例化函数对象,我们也可以通过函数对象所属类进行一个函数对象的实例化。

2、两个函数对象可以指向通过一个字节码对象,表示执行相同的计算逻辑。

3、通过def关键字实例化的函数对象的__globals__属性,即为函数定义当前的全局命名空间对应的字典对象,无法修改。

4、通过手动实例化函数对象的方式,我们可以通过globals关键字传参,来手工指定函数对象的__globals__属性,从而实现,函数对象的计算逻辑在不同的命名空间中执行的效果。


函数对象中还有一个__call__()方法,这个方法之前其实已经介绍过,我们调用函数,其实有两种方式,一种是函数名()来进行调用。另外一种可以通过函数名.__call__()的方式进行调用。

这两种方式本质上是一样的。就不再展开了,感兴趣的同学可以自行尝试。


函数对象中的代码在哪里

通过前面内容的介绍,我们可以发现,函数对象的核心在于存储计算逻辑的__code__属性,以及存储函数执行的上下文环境对应的__globals__属性。两者的不同组合,可以实现函数对象的实际执行的多样性。

接下来我们来进一步看一下存储函数对象代码的__code__属性。

首先看代码:

a = 20


def add(n=1000, *, m=23000):
    b = 10
    c = 20
    return a + n


code = add.__code__
print(code.__class__)
print(code.co_name)
print(code.co_argcount)
print(code.co_kwonlyargcount)
print(code.co_nlocals)
print(code.co_varnames)
print(code.co_names)
print(code.co_consts)
print(code.co_stacksize)
print(code.co_code)

执行结果:

简单解释一下code对象的相关属性:

1、co_name:存储函数的名字,通常情况下与函数对象的__name__相同。

2、co_argcount:存储函数的位置参数的个数,代码示例中只有一个n。

3、co_kwonlyargcount:存储函数的强制关键字传参的参数个数,代码中只有一个m。

4、co_nlocals:存储函数中的局部变量的个数,代码示例中为4个,局部变量包括函数的参数及函数内部定义的局部变量。

5、co_varnames:存储函数中的局部变量的名称的元组,这里是n, m, b, c。

6、co_names:存储函数中使用的全局变量的名称的元组,这里只有a。

7、co_consts:存储函数中使用的常量的元组,这里有None、10、20。

8、co_stacksize:存储函数执行所需要的栈空间,这里是2。

9、co_code:存储函数执行的代码所编译为的字节码。


总结

本文重点介绍了函数对象中的比较重要的属性,以及通过函数类进行函数对象的实例化,然后展开介绍了函数属性中的__code__属性中的相关属性,从而对Python中的函数对象有了更深入的理解。


感谢您的拨冗阅读,希望对您学习Python能有些许帮助。

相关推荐

hdd硬盘和ssd(ssd硬盘和hdd硬盘是什么意思)

HDD硬盘和SSD硬盘是两种不同类型的电脑存储设备,它们有着以下区别:1.工作原理:HDD硬盘使用机械旋转的磁盘和读写磁头来存储和读取数据,而SSD硬盘则使用闪存存储数据,类似于USB闪存盘。2....

电脑免费软件下载大全(电脑上免费的下载软件)

正常情况下,如果我们想要在自己的电脑上面下载一个不要钱的单机游戏,那么我们是可以直接在我们的软件管理中心进行一个下载的,这个时候我们只需要通过一个权限就能够正常的下载,当然我们也是可以在一些小游戏的软...

mpp文件转换excel(mpp转换成pdf)

要将Excel表格转换为MPP格式,您可以按照以下步骤操作:1.打开Excel表格并确保数据按照项目的不同阶段或任务进行组织。2.将Excel表格中的数据复制到一个新的MicrosoftProj...

win7旗舰版开机密码忘记按f2

方法如下:开始-控制面板-用户帐户;在打开的更改用户帐户界面点击要更改的帐户;然后点击帐户左面的更改密码按钮;在打开的页面上,输入一次当前使用的密码,输入2次要更改的新密码然后保存退出就可以了...

笔记本无音频输出设备(笔记本无音频输出设备)

1、没有声卡驱动,解决方法就是找到笔记本的官网,下载电脑声卡的驱动安装即可。2、没有外界的音频播放设备,解决方法就是买一个外界的音频播放设备插到电脑主机的音频接口上即可。笔记本电脑显示未安装任何音频输...

iso文件能用手机打开吗(iso文件能用手机打开吗安全吗)

一般的压缩软件就可以打开的,比如,好压软件,这个打开只是解压形式的,如果你说的是运行iso文件,这个没有,况且安卓系统也不支持iso运行ISO文件一般用于光盘镜像文件的存储,如果想要在手机上运行ISO...

win7系统卡顿怎么优化(win7很慢很卡怎么优化)

1、首先打开安全卫士,进入安全卫士首页,单击软件窗口右下角的“更多”图标,打开扩展应用程序。2、单击选择“我的工具”。3、在我的工具菜单里面找到“人工服务”单击打开人工服务。4、在人工服务对话框有很多...

如何查看c盘微信聊天记录(如何查看c盘微信聊天记录内存大小)

微信群中的消息只要没删除基本都能保存,想要找微信群中几个多月前的消息可以直接根据日期来查找聊天记录。操作如下:1、打开想要查找记录的微信群,点击右上角人形图标;2、点击查找聊天内容;3、选择按日...

office2016家庭版激活密钥(office家庭版激活码2019)

走淘宝吧,因为零售版的密钥只能用一次。大概几块钱就能激活2016。如果你不在乎钱的话可以向我一样,订阅一个office365.实在不行可以和几个人一起买一个家庭版的365.出现这个情况,找微软申诉是没...

移动硬盘驱动器下载安装(移动硬盘驱动器下载安装教程)

1、右键单击您的桌面,选择“新建文件夹”,并命名该文件夹(例如“usb驱动程序”);2、然后到本站下载驱动程序;3、将其解压缩至在您的桌面上刚刚创建的usb驱动程序文件夹;4、单击开始菜单,然后选择设...

电脑硬盘格式化工具(电脑 格式化硬盘)

硬盘格式化工具很多,PQMACGIG8.0(中文就叫硬盘分区魔法师)是比较好的一个,这个是在WINDOWS下比叫好用,(个人感觉)FDISK也是比较好的一个,这个一般用在DOS下分区格式化WIN...

photoshop是一款什么软件(ps指的是什么软件)

这个说法是错误的,ps软件“即:photoshop”是由美国著名的“adobe阿多比”公司出品的专业的图像处理软件,它不是由微软公司出品的软件。众所周知的是,微软公司以设计视窗操作系统名满全球,它出...

ipad越狱的好处与坏处(ipad越狱好不好)

  好处一:  1、重命名、重组应用程序  如果你看着Sparrow(iOS最优秀邮件客户端)这个名字不爽,越狱之后就可以改成“Email”,如果你觉得“豆瓣电台”这个名字不给力,那就改成“中央人民广...

win7光盘重装系统步骤图解(win7光盘如何重装系统)

1.确认您的电脑支持从光盘启动。如果支持,可以直接将Windows7安装光盘插入电脑的光驱中。 2.打开电脑,按下F2、F10、F12或Delete等键进入BIOS设置界面。 ...

电脑已联网却无法上网(电脑已经联网了但是不能上网)

电脑连上网后,仍可能存在无法上网的情况,这可能是由多种原因造成的。以下是一些可能的原因和解决方法:1.浏览器问题:有时候,浏览器可能会出现故障,导致无法正常访问网络。您可以尝试清除浏览器的缓存和co...

取消回复欢迎 发表评论: