返回介绍

Python 命名空间的禅:赋值将变量名分类

发布于 2024-01-29 22:24:15 字数 1681 浏览 0 评论 0 收藏 0

点号和无点号的变量名有不同的搜索流程,再加上两者都有多个搜索层次,有时很难看出变量名最终属于何处。在Python中,赋值变量名的场所相当重要:这完全决定了变量名所在的作用域或对象。文件manynames.py示范了这条原则是如何变成代码的,并总结了本书遇到的命名空间的概念。

这个文件分别五次给相同的变量名X赋值。不过,因为这个名称是在五个不同地方进行赋值的,这个程序中的五个X是完全不同的变量。从上至下,这里对X的赋值语句会产生:模块属性(11)、函数内的本地变量(22)、类属性(33)、方法中的本地变量(44)以及实例属性(55)。虽然这五个都称为X,但事实上它们都是在原代码内的不同位置进行赋值的,或者说是赋值到了不同的对象,因此,使得这些变量名都是独特的变量。

你应该花时间仔细研究一下这个例子,因为这个例子把本书前几部分探索过的概念都集合起来了。当你看懂时,就完成了Python命名空间的涅槃重生。当然,另一条达到涅槃状态的途径就是运行这个程序,看看运行结果是什么。以下是这个源代码的其余部分,也就是制作实例,打印其能读取的所有的X。

文件执行时所打印的输出就在程序代码的注释中。可以跟踪这些输出来了解每次读取的变量X是哪一个。注意:可以通过类来读取其属性(C.X),但无法从def语句外读取函数或方法内的局部变量。局部变量对于在def内的其余代码才是可见的。而事实上,也只有当函数调用或方法执行时,才会存在于内存中。

这个文件定义的其中一些变量名可以让文件以外的其他模块看见。但是,回想一下,我们在另一个文件内读取这些变量名前,总是需要先进行导入。这就是模块的重点所在。

注意:manynames.f()是怎样打印manynames中的X的,而不是打印本文件中赋值的X。作用域总是由源代码中的赋值语句位置来决定的(也就是语句),而且绝不会受到其导入关系的影响。此外,直到我们调用I.m()前实例的X都不会创建:属性就像是变量,在赋值之后才会存在,而不是在赋值前。在通常情况下,创建实例属性的方法是在类的__init__构造函数内进行赋值的,但这并不是唯一的选择。

最后,正如我们在第17章所了解到的,一个函数在其外部修改名称也是可能的,使用global和(Python 3.0中的)nonlocal语句——这些语句提供了写入访问,但是也修改了赋值的命名空间绑定规则:

当然,通常来说,在脚本内每个变量都不应该使用相同的变量名!但是,就像这个例子所表示的那样,即使这么做,Python的命名空间还是会工作,防止在一个环境中所用的变量名无意中和另一个环境中所使用的变量名发生冲突。

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文