Python中使用语句导入模块或包的机制研究

作者:Shahriar Tajbakhsh 时间:2023-02-21 15:06:40 

这篇文章讨论了Python的from <module> import *和from <package> import *,它们怎么执行以及为什么使用这种语法(也许)是一个坏主意。
从一个模块导入全部

from <module> import * means意味着“我希望能访问<module>中我有权限访问的全部名称”。例如以下代码something.py:
 


# something.py

public_variable = 42
_private_variable = 141

def public_function():
 print("I'm a public function! yay!")

def _private_function():
 print("Ain't nobody accessing me from another module...usually")

class PublicClass(object):
 pass

class _WeirdClass(object):
 pass

在Python解释器中,我们可以执行from something import *,然后看到如下的内容:
 


>>> from something import *
>>> public_variable
42
>>> _private_variable
...
NameError: name '_private_variable' is not defined
>>> public_function()
"I'm a public function! yay!"
>>> _private_function()
...
NameError: name '_private_function' is not defined
>>> c = PublicClass()
>>> c
<something.PublicClass object at ...>
>>> c = _WeirdClass()
...
NameError: name '_WeirdClass' is not defined

from something import *从something中导入了除了以_开头名称外的其他所有名称,按照规范,_开始的名称是私有的所以未被导入。
嗯,不是特别糟!还有什么?

上面没提到__all__是什么。__all__是一个字符串列表,指定了当from <module> import *被使用时,模块(或者如后文会提到的包)中的哪些符号会被导出。如果我们不定义__all__(我们在上面的something.py就没定义),import *默认的导入方式是导入除了下划线(_)开头的所有名称。再说一次,编程惯例上下划线表示一个符号是私有的,不导入是合理的。让我们来看看在something.py中定义我们自己的__all__会发生什么。
 


# something.py

__all__ = ['_private_variable', 'PublicClass']

# The rest is the same as before

public_variable = 42
_private_variable = 141

def public_function():
 print("I'm a public function! yay!")

def _private_function():
 print("Ain't nobody accessing me from another module...usually")

class PublicClass(object):
 pass

class _WeirdClass(object):
 pass

现在,我们期望from something import *只会导入_private_variable和PublicClass:
 


>>> from something import *
>>> public_variable
42
>>> _private_variable
...
NameError: name '_private_variable' is not defined
>>> public_function()
"I'm a public function! yay!"
>>> _private_function()
...
NameError: name '_private_function' is not defined
>>> c = PublicClass()
>>> c
<something.PublicClass object at ...>
>>> c = _WeirdClass()
...
NameError: name '_WeirdClass' is not defined

包是怎样的呢?

当从一个包中导入全部时,__all__的做法和模块基本一样,不过它处理的是包中的模块(而不是把模块中的名都导入)。所以当我们使用from <package> import *.时__all__说明了所有需要被导入当前命名空间的模块。

不同之处在于,如果你在一个包的__init__.py里面没有声明__all__,from <package> import *语句不会导入任何东西(这个说法也不全对,正确的说法在此)
但是,这有什么不好?

继续读之前,在你的Python解释器中,执行import this,再读一遍Python之禅(在你孩子每晚睡前也要读给他们)。

    明确比含糊要好。

from <module> import * 是不明确的。它没告诉我们我们正在导入什么或者我们把什么带入当前命名空间了。更好的做法是显式地导入我们需要的全部名称。这种方式下,读者(非常可能是未来的你自己)就不会困惑于你代码中使用的一个变量/方法/类/其他东西是哪儿来的,这也告诉了我们下一点:

    可读性很重要

即使你需要导入很多东西,一个一个显式地导入也更清楚。使用PEP 328:
 


from Tkinter import (Tk, Frame, Button, Entry, Canvas, Text,
 LEFT, DISABLED, NORMAL, RIDGE, END)

你现在就能明确知道你的命名空间里有什么,使用ctrl+f能很快地告诉你它们是哪儿来的。

同时,你还总是要承担模块/包作者更改list内容(加/减东西)的风险。也就是下面两者之一:

    作者从__all__里删除了一个字符串。如果你的代码使用了那个名字,你的代码就会报出NameError的错误,并且很难发现为什么。
    作者在__all__里加入了很多东西。你也许不需要这些增加的内容,所以你只是让这些你不关心的东西占满了你的命名空间。他们甚至在你不注意的时候会替代其他同名内容。

当然,有时候从模块或者包中导入全部内容是有用的。不过,这么做之前三思。从我的经验来看,这么做通常只是因为懒。

标签:Python,导入包,导入模块
0
投稿

猜你喜欢

  • 手写Vue2.0 数据劫持的示例

    2024-05-22 10:43:17
  • Python中列表、字典、元组、集合数据结构整理

    2022-08-02 05:54:57
  • python pandas 时间日期的处理实现

    2021-09-05 02:38:07
  • Django Path转换器自定义及正则代码实例

    2022-05-19 09:01:25
  • PyQt5 实现字体大小自适应分辨率的方法

    2022-08-25 03:07:44
  • 一个统计表每天的新增行数及新增存储空间的功能

    2024-01-13 22:27:40
  • python文件读写操作与linux shell变量命令交互执行的方法

    2022-10-24 06:37:55
  • 浅谈FastClick 填坑及源码解析

    2024-04-10 16:08:57
  • Python基础学习之反射机制详解

    2023-02-15 11:02:01
  • VSCode插件安装完成后的配置详解

    2022-02-16 00:36:34
  • MySQL开发规范与使用技巧总结

    2024-01-19 00:44:00
  • python人工智能算法之线性回归实例

    2023-09-12 10:14:52
  • MySQL增删查改数据表详解

    2024-01-25 06:55:48
  • Python设计模式编程中的备忘录模式与对象池模式示例

    2023-02-06 05:48:43
  • python中文文本切词Kmeans聚类

    2023-06-01 18:35:57
  • 解决Vue不能检测数组或对象变动的问题

    2024-04-27 15:59:40
  • python写入文件自动换行问题的方法

    2022-06-13 11:28:18
  • Python函数定义及传参方式详解(4种)

    2022-08-16 14:39:08
  • pandas 按照特定顺序输出的实现代码

    2023-02-04 19:35:17
  • 如何通过Vue实现@人的功能

    2024-06-05 15:32:04
  • asp之家 网络编程 m.aspxhome.com