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


Posted in Python onMarch 30, 2015

这篇文章讨论了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 相关文章推荐
python self,cls,decorator的理解
Jul 13 Python
python实现人脸识别代码
Nov 08 Python
浅谈pandas中shift和diff函数关系
Apr 08 Python
Python面向对象类的继承实例详解
Jun 27 Python
python创建文件备份的脚本
Sep 11 Python
CentOS 7 安装python3.7.1的方法及注意事项
Nov 01 Python
对Python之gzip文件读写的方法详解
Feb 08 Python
六行python代码的爱心曲线详解
May 17 Python
如何在python中写hive脚本
Nov 08 Python
使用PyQt5实现图片查看器的示例代码
Apr 21 Python
django orm模块中的 is_delete用法
May 20 Python
Python改变对象的字符串显示的方法
Aug 01 Python
优化Python代码使其加快作用域内的查找
Mar 30 #Python
Python中分数的相关使用教程
Mar 30 #Python
Python2.x中str与unicode相关问题的解决方法
Mar 30 #Python
分享一个常用的Python模拟登陆类
Mar 29 #Python
python实现查询IP地址所在地
Mar 29 #Python
python实现定时播放mp3
Mar 29 #Python
Python实现设置windows桌面壁纸代码分享
Mar 28 #Python
You might like
php垃圾代码优化操作代码
2010/08/05 PHP
PHP正确配置mysql(apache环境)
2011/08/28 PHP
PHP数组遍历知识汇总(包含遍历方法、数组指针操作函数、数组遍历测速)
2014/07/05 PHP
PHP Yii框架之表单验证规则大全
2015/11/16 PHP
php用户登录之cookie信息安全分析
2016/05/13 PHP
Yii2使用自带的UploadedFile实现的文件上传
2016/06/20 PHP
详解EventDispatcher事件分发组件
2016/12/25 PHP
php中Ioc(控制反转)和Di(依赖注入)
2017/05/07 PHP
javascript EXCEL 操作类代码
2009/07/30 Javascript
js下获取div中的数据的原理分析
2010/04/07 Javascript
javascript之typeof、instanceof操作符使用探讨
2013/05/19 Javascript
Jquery chosen动态设置值实例介绍
2013/08/08 Javascript
浅谈Node.js中的定时器
2015/06/18 Javascript
JS实现可点击展开与关闭的左侧广告代码
2015/09/02 Javascript
JavaScript计划任务后台运行的方法
2015/12/18 Javascript
很酷的星级评分系统原生JS实现
2016/08/25 Javascript
JavaScript算法系列之快速排序(Quicksort)算法实例详解
2016/09/04 Javascript
谈谈对JavaScript原生拖放的深入理解
2016/09/20 Javascript
原生js仿浏览器滚动条效果
2017/03/02 Javascript
angular $watch 一个变量的变化(实例讲解)
2017/08/02 Javascript
vue2导航根据路由传值,而改变导航内容的实例
2017/11/10 Javascript
使用sklearn之LabelEncoder将Label标准化的方法
2018/07/11 Python
Flask之flask-script模块使用
2018/07/26 Python
python list格式数据excel导出方法
2018/10/31 Python
使用python制作一个为hex文件增加版本号的脚本实例
2019/06/12 Python
关于keras.layers.Conv1D的kernel_size参数使用介绍
2020/05/22 Python
Pandas读取csv时如何设置列名
2020/06/02 Python
如何通过python实现IOU计算代码实例
2020/11/02 Python
Django多个app urls配置代码实例
2020/11/26 Python
《彭德怀和他的大黑骡子》教学反思
2014/04/12 职场文书
法人委托书的范本格式
2014/09/11 职场文书
离婚协议书怎么写的
2014/12/14 职场文书
小学班主任工作随笔
2015/08/15 职场文书
四则混合运算教学反思
2016/02/23 职场文书
详解Go语言运用广度优先搜索走迷宫
2021/06/23 Python
分析MySQL优化 index merge 后引起的死锁
2022/04/19 MySQL