Python __all__ 的作用与限制

Python __all__ 的作用与限制

在 Python 中,__all__ 是一个特殊的模块级变量,它主要用于定义模块或包的公开接口,即哪些模块或函数可以通过 from <module> import * 的方式导入。了解 __all__ 的工作原理和限制可以帮助我们更好地组织和管理代码的可见性。

1. __all__ 的基本作用

1.1 定义模块或包的公开接口

__all__ 控制的是 from <module> import * 导入时的内容,它用于指定哪些函数、类或变量可以被导入到当前命名空间。如果你不显式地定义 __all__,Python 会导入模块中所有没有以下划线 (_) 开头的成员。

1.2 控制 import * 导入行为

  • 如果一个模块定义了 __all__,当使用 from <module> import * 时,只有 __all__ 中列出的成员会被导入。
  • 如果没有定义 __all__import * 会导入模块中所有公共的(不以 _ 开头的)成员。
python 复制代码
# my_module1.py
def fun01():
    print("fun01 called")

def fun02():
    print("fun02 called")

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

__all__ = ['fun01', 'fun02']  # 仅暴露这两个函数
python 复制代码
# other_module.py
from my_module1 import *  # 只会导入 fun01 和 fun02

fun01()  # works
fun02()  # works
get_sum(1, 2)  # 会报错,因为 get_sum 不在 __all__ 中

在上面的代码中,get_sum 没有在 __all__ 中,因此它不会通过 import * 导入,而 fun01fun02 会被导入。

2. __all__ 不会限制通过模块名访问

尽管 __all__ 可以限制 from <module> import * 导入时的内容,但它 不会影响通过模块名直接访问模块的功能

2.1 模块本身的访问

如果你使用 import <module> as <alias> 的方式导入模块,则 模块中的所有成员 都可以通过模块别名访问,无论这些成员是否在 __all__ 中列出。

python 复制代码
# other_module.py
from my_module1 import my_module1 as m1

m1.fun01()  # works, fun01 是公开的
m1.fun02()  # works, fun02 是公开的
print(m1.get_sum(1, 2))  # works, 即使 get_sum 不在 __all__ 中,仍然可以通过 m1 调用

在这个例子中,虽然 get_sum 没有出现在 __all__ 中,但你仍然可以通过 m1.get_sum() 来访问它。

2.2 为什么 get_sum 仍然能被调用?

__all__ 控制的是 import * 的行为,它不会影响 通过模块对象访问模块内容的行为 。当你通过 import my_module1 as m1 导入模块时,m1 代表整个模块,你可以访问模块中的所有非私有成员,无论它们是否在 __all__ 中。

3. 使用 __all__ 的实际意义

3.1 控制包的公共接口

在一个包的 __init__.py 文件中,定义 __all__ 使得包对外暴露特定的模块或函数。这有助于避免导入不需要的模块或函数,保持代码的模块化和简洁性。

ini 复制代码
# package/__init__.py
__all__ = ['module1', 'module2']  # 只公开 module1 和 module2

3.2 提高代码可维护性

通过显式定义 __all__,可以确保包的公共接口清晰明确,其他开发者在使用这个包时,只会看到你希望公开的部分,避免暴露内部实现的细节。

3.3 让包更易于使用

当你定义了 __all__ 后,使用者只需要 from package import * 即可导入所需的功能,避免了冗长的导入语句,提升了包的易用性。

4. 总结

4.1 知识点总结

  • __all__ 控制的是 from <module> import * 导入时的行为,定义哪些模块、类或函数可以被导入。
  • __all__ 不会影响 通过模块名直接访问 模块时的可访问性,所有模块的成员都可以通过模块对象访问。
  • __all__ 提供了一个明确的公共接口,帮助避免导入不必要的模块或暴露不希望公开的内部实现。

通过理解和合理使用 __all__,你可以更好地组织和管理 Python 项目的模块化和封装性。

4.2 最后总结

这是我学习 Python 后的第一篇文章,也是我人生的第一篇文章。如果有不足之处,欢迎讨论交流。希望这篇文章对你有所帮助。如果你觉得有帮助,欢迎关注、点赞、收藏,感谢支持!

Bye~

相关推荐
带娃的IT创业者1 小时前
《Python实战进阶》No39:模型部署——TensorFlow Serving 与 ONNX
pytorch·python·tensorflow·持续部署
Bruce-li__1 小时前
深入理解Python asyncio:从入门到实战,掌握异步编程精髓
网络·数据库·python
九月镇灵将1 小时前
6.git项目实现变更拉取与上传
git·python·scrapy·scrapyd·gitpython·gerapy
小张学Python2 小时前
AI数字人Heygem:口播与唇形同步的福音,无需docker,无需配置环境,一键整合包来了
python·数字人·heygem
跳跳糖炒酸奶2 小时前
第四章、Isaacsim在GUI中构建机器人(2):组装一个简单的机器人
人工智能·python·算法·ubuntu·机器人
步木木2 小时前
Anaconda和Pycharm的区别,以及如何选择两者
ide·python·pycharm
星始流年2 小时前
解决PyInstaller打包PySide6+QML应用的资源文件问题
python·llm·pyspider
南玖yy2 小时前
Python网络爬虫:从入门到实践
爬虫·python
The Future is mine3 小时前
Python计算经纬度两点之间距离
开发语言·python
九月镇灵将3 小时前
GitPython库快速应用入门
git·python·gitpython